Compare commits

...

69 Commits

Author SHA1 Message Date
Spythere f53f3a18fe Merge pull request #109 from Spythere/development
v1.28.0
2024-09-09 14:26:32 +02:00
Spythere fac8fced3e chore: removed journal list status animations 2024-09-09 14:11:35 +02:00
Spythere a3d9e68c8a chore: tooltip placing 2024-09-08 16:32:11 +02:00
Spythere b09761de58 chore: TWR & SKR badges fixes 2024-09-07 21:07:41 +02:00
Spythere 8ac2c68660 bump: v1.28.0 2024-09-07 17:28:29 +02:00
Spythere 4177c6e5f4 chore: displaying warning notes in driver view & journal timetables 2024-09-07 17:28:05 +02:00
Spythere b8f135a454 chore: thumbnail loading optimization 2024-09-06 15:23:22 +02:00
Spythere f0863b2459 chore: added static data hourly refresh 2024-09-05 17:00:15 +02:00
Spythere 55b4732992 chore: cleanup 2024-09-05 15:34:11 +02:00
Spythere 7b3dcea89e fix: missing category translations 2024-09-03 15:25:10 +02:00
Spythere f4b0c39185 chore: entry details backwards comp. 2024-09-03 15:18:59 +02:00
Spythere 275d602f97 chore: hiding entry details on history change 2024-09-03 14:43:16 +02:00
Spythere c93514fdf0 refactor: journal timetable entries 2024-09-03 14:29:59 +02:00
Spythere 0861d92e4b hotfix: class name 2024-09-02 23:41:39 +02:00
Spythere bdfd73f4be chore: stops design 2024-09-02 22:56:00 +02:00
Spythere df86364c51 chore: journal timetable stop labels 2024-09-02 22:39:41 +02:00
Spythere 631bb20c61 hotfix: resolved merge conflicts 2024-09-01 16:09:16 +02:00
Spythere bed79ed2d0 chore: translations 2024-09-01 16:07:40 +02:00
Spythere 2a07471e12 chore: displaying other driver's trains in the driver view 2024-08-31 13:51:22 +02:00
Spythere cfe188d0dc bump: v1.27.1 2024-08-29 16:04:14 +02:00
Spythere d9865be83e fix: views viewport height 2024-08-29 16:03:26 +02:00
Spythere 9155fd9f8d chore: driver view return button 2024-08-29 15:51:53 +02:00
Spythere 4674bf886e fix: manifest theme color 2024-08-29 15:47:23 +02:00
Spythere 289fd310df chore: viewport & routing fixes 2024-08-29 15:42:44 +02:00
Spythere b04797052f chore: expanded containers width in views, adjusted dropdowns 2024-08-24 16:16:32 +02:00
Spythere 7079f20791 Merge pull request #107 from Spythere/development
hotfix: journal stats badge styles
2024-08-24 00:25:35 +02:00
Spythere c244275aee hotfix: journal stats badge styles 2024-08-24 00:24:55 +02:00
Spythere fbc9785341 Merge pull request #105 from Spythere/development
v1.27.0
2024-08-24 00:20:08 +02:00
Spythere 9fd02c2336 chore: scenery view border radius 2024-08-23 16:24:21 +02:00
Spythere c031dd55c1 chore: removed return button 2024-08-23 16:22:54 +02:00
Spythere b0870699a4 hotfix: anchor style 2024-08-22 23:20:09 +02:00
Spythere a8da634b0e hotfix: TrainsView watcher causing routing problems 2024-08-22 23:19:42 +02:00
Spythere 8920b1e5e8 hotfix: view styles 2024-08-22 17:00:40 +02:00
Spythere 4fa1c05831 fix: input attrs 2024-08-22 16:44:51 +02:00
Spythere ecef2d5ee4 chore: backwards compatibility with train modal for ext. links 2024-08-22 16:37:47 +02:00
Spythere 1749871d08 fix: filters double click 2024-08-22 16:37:12 +02:00
Spythere 5545616706 chore: styling hotfixes and improvements 2024-08-22 02:28:40 +02:00
Spythere b35bb03868 merge: 'dominik-korsa-links' into development 2024-08-22 02:27:53 +02:00
Spythere b9521918cb hotfix: translation fix 2024-08-21 21:38:21 +02:00
Spythere 7ad17fc2c5 chore: style improvements & finishing touches 2024-08-21 21:33:46 +02:00
Spythere a80144cb1c chore: missing translations 2024-08-21 18:30:17 +02:00
Spythere 1227cdb94a chore: driver view links 2024-08-21 18:12:35 +02:00
Spythere b33594fd6f fix: journal list styles 2024-08-21 17:17:52 +02:00
Spythere 9f8656e590 fix: progress indicator 2024-08-21 17:06:50 +02:00
dominik-korsa 81cd165fe7 Use <router-link> instead of <tr> with click handler in StationTable 2024-08-21 14:42:03 +02:00
dominik-korsa 41e4b45599 Use <router-link> for driver journal button in TrainInfo 2024-08-21 13:54:48 +02:00
dominik-korsa 462dd7dd7a Replace all remaining uses of driverViewMixin with <router-link> 2024-08-21 13:49:31 +02:00
dominik-korsa 9837ae97e1 Use <router-link> instead of a @click handler in SceneryTimetable 2024-08-21 13:29:53 +02:00
dominik-korsa a818cd980b Use <router-link> instead of a @click handler in TrainTable 2024-08-21 12:58:47 +02:00
Spythere 573ebc233b chore: removed registering train modal 2024-08-21 02:06:38 +02:00
Spythere aae47c6abd fix: journal stats badges 2024-08-21 02:06:23 +02:00
Spythere 24c9b62162 feat: driver train view 2024-08-21 02:02:35 +02:00
Spythere 481d43b6d8 chore: selecting station checkpoint from url 2024-08-20 14:31:52 +02:00
Spythere 4969a433cc fix: train modal responsiveness & icons 2024-08-20 13:33:24 +02:00
Spythere 8a2b453dc6 chore: journal daily stats styling 2024-08-20 13:16:37 +02:00
Spythere 86d178ef56 chore: restored Pragotron link 2024-08-20 13:16:02 +02:00
Spythere 7769477508 chore: station stats styling 2024-08-20 00:14:44 +02:00
Spythere 551b60c733 fix: omitting "po" stops in timetable progress bar 2024-08-19 23:56:15 +02:00
Spythere 80a5b56785 feat: router links embeded into timetable stop names 2024-08-18 23:45:42 +02:00
Spythere 6bd62f13a1 chore: stats button responsiveness 2024-08-18 23:14:18 +02:00
Spythere 42591f6e76 chore: journal timetables styling improvements 2024-08-18 23:03:00 +02:00
Spythere 4ca0f09e75 chore: added ZG category 2024-08-18 22:58:22 +02:00
Spythere 02c3629c00 bump: v1.27.0 2024-08-18 01:43:51 +02:00
Spythere 9c4c806f0e chore: moved station stats to a dropdown 2024-08-18 01:43:27 +02:00
Spythere 58d6a97762 fix: twr/skr badges sizing 2024-08-17 15:51:36 +02:00
Spythere cd71c78eb4 chore: apicache typings 2024-08-17 15:51:25 +02:00
Spythere 300e70dcfe chore: journal timetables buttons alignment 2024-08-15 15:01:36 +02:00
Spythere 09b31f7914 Merge pull request #104 from Spythere/development
hotfix: warning icon placement
2024-08-10 23:41:34 +02:00
Spythere c29c3c6abe hotfix: warning icon placement 2024-08-10 23:41:00 +02:00
54 changed files with 1560 additions and 1094 deletions
+1 -1
View File
@@ -1,6 +1,6 @@
{ {
"name": "stacjownik", "name": "stacjownik",
"version": "1.26.1", "version": "1.28.0",
"private": true, "private": true,
"type": "module", "type": "module",
"scripts": { "scripts": {
+1 -1
View File
@@ -13,7 +13,7 @@
"type": "image/png" "type": "image/png"
} }
], ],
"theme_color": "#ffc014", "theme_color": "#4d4d4d",
"background_color": "#4d4d4d", "background_color": "#4d4d4d",
"display": "standalone", "display": "standalone",
"start_url": "." "start_url": "."
-9
View File
@@ -6,13 +6,6 @@
/> />
<Tooltip /> <Tooltip />
<transition name="modal-anim">
<keep-alive>
<TrainModal />
</keep-alive>
</transition>
<AppHeader :current-lang="currentLang" @change-lang="changeLang" /> <AppHeader :current-lang="currentLang" @change-lang="changeLang" />
<main class="app_main"> <main class="app_main">
@@ -54,7 +47,6 @@ import { useTooltipStore } from './store/tooltipStore';
import Clock from './components/App/Clock.vue'; import Clock from './components/App/Clock.vue';
import StatusIndicator from './components/App/StatusIndicator.vue'; import StatusIndicator from './components/App/StatusIndicator.vue';
import AppHeader from './components/App/AppHeader.vue'; import AppHeader from './components/App/AppHeader.vue';
import TrainModal from './components/TrainsView/TrainModal.vue';
import Tooltip from './components/Tooltip/Tooltip.vue'; import Tooltip from './components/Tooltip/Tooltip.vue';
import UpdateCard from './components/App/UpdateCard.vue'; import UpdateCard from './components/App/UpdateCard.vue';
@@ -67,7 +59,6 @@ export default defineComponent({
Clock, Clock,
StatusIndicator, StatusIndicator,
AppHeader, AppHeader,
TrainModal,
UpdateCard, UpdateCard,
Tooltip Tooltip
}, },
-1
View File
@@ -43,7 +43,6 @@ export default defineComponent({
width: 6em; width: 6em;
height: 1em; height: 1em;
margin: 0.5em 0;
.bar-fg, .bar-fg,
.bar-bg { .bar-bg {
+2 -12
View File
@@ -8,7 +8,7 @@
:key="i" :key="i"
> >
<div class="stock-text"> <div class="stock-text">
<p>{{ vehicleName.replace(/_/g, ' ') }}</p> <div>{{ vehicleName.replace(/_/g, ' ') }}</div>
<small v-if="vehicleCargo">({{ vehicleCargo }})</small> <small v-if="vehicleCargo">({{ vehicleCargo }})</small>
</div> </div>
@@ -180,7 +180,6 @@ export default defineComponent({
align-items: flex-end; align-items: flex-end;
overflow: auto; overflow: auto;
margin: 0 auto; margin: 0 auto;
padding: 1em 0;
} }
ul > li > span { ul > li > span {
@@ -189,20 +188,11 @@ ul > li > span {
cursor: crosshair; cursor: crosshair;
} }
img {
max-height: 60px;
width: auto;
height: auto;
}
img.traction-only {
max-width: 100%;
}
.stock-text { .stock-text {
text-align: center; text-align: center;
color: #aaa; color: #aaa;
font-size: 0.9em; font-size: 0.9em;
margin-bottom: 0.25em; margin-bottom: 0.25em;
padding: 0.25em 0;
} }
</style> </style>
+8 -11
View File
@@ -1,5 +1,5 @@
<template> <template>
<div class="vehicle-thumbnail"> <div class="vehicle-thumbnail" :data-load-status="imgStatus">
<img <img
ref="imgRef" ref="imgRef"
:src="`https://static.spythere.eu/thumbnails/v2/${imgName}.png`" :src="`https://static.spythere.eu/thumbnails/v2/${imgName}.png`"
@@ -7,7 +7,6 @@
loading="lazy" loading="lazy"
data-tooltip-type="VehiclePreviewTooltip" data-tooltip-type="VehiclePreviewTooltip"
:data-tooltip-content="vehicleName" :data-tooltip-content="vehicleName"
:data-load-status="imgStatus"
@error="onImageError" @error="onImageError"
@load="onImageLoad" @load="onImageLoad"
/> />
@@ -15,7 +14,7 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { onMounted, Ref, ref } from 'vue'; import { Ref, ref } from 'vue';
const props = defineProps({ const props = defineProps({
vehicleName: { type: String, required: true }, vehicleName: { type: String, required: true },
@@ -29,8 +28,6 @@ const imgRef = ref(null) as Ref<HTMLElement | null>;
const imgStatus = ref('loading'); const imgStatus = ref('loading');
function onImageError(event: Event) { function onImageError(event: Event) {
console.log('error');
(event.target as HTMLImageElement).src = `/images/${props.fallbackName}.png`; (event.target as HTMLImageElement).src = `/images/${props.fallbackName}.png`;
imgStatus.value = 'error'; imgStatus.value = 'error';
} }
@@ -40,22 +37,22 @@ function onImageLoad() {
imgStatus.value = 'loaded'; imgStatus.value = 'loaded';
} }
imgRef.value!.style.opacity = '1'; if (imgRef.value) imgRef.value.style.opacity = '1';
} }
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
.vehicle-thumbnail { .vehicle-thumbnail {
position: relative; position: relative;
&[data-load-status='loading'] {
min-height: 60px;
min-width: 200px;
}
} }
img { img {
opacity: 0; opacity: 0;
transition: opacity 100ms ease-in-out; transition: opacity 100ms ease-in-out;
&[data-load-status='loading'] {
min-height: 60px;
min-width: 150px;
}
} }
</style> </style>
@@ -1,29 +1,28 @@
<template> <template>
<section class="daily-stats"> <section class="daily-stats">
<span :data-active="statsStatus"> <span :data-active="statsStatus">
<span class="stats-list"> <h3>
<h3> {{ $t('journal.daily-stats.title') }}
{{ $t('journal.daily-stats.title') }} <b class="text--primary">{{ new Date().toLocaleDateString($i18n.locale) }}</b>
<b class="text--primary">{{ new Date().toLocaleDateString($i18n.locale) }}</b> </h3>
</h3>
<hr class="header-separator" /> <hr class="header-separator" />
<b v-if="statsStatus == Status.Data.Loading"> <b v-if="statsStatus == Status.Data.Loading">
{{ $t('app.loading') }} {{ $t('app.loading') }}
</b> </b>
<b class="text--error" v-else-if="statsStatus == Status.Data.Error"> <b class="text--error" v-else-if="statsStatus == Status.Data.Error">
{{ $t('journal.stats-error') }} {{ $t('journal.stats-error') }}
</b> </b>
<b v-else-if="topDispatchers.length == 0"> <b v-else-if="topDispatchers.length == 0">
{{ $t('journal.daily-stats.info') }} {{ $t('journal.daily-stats.info') }}
</b> </b>
<div v-else> <div v-else>
<div v-if="stats.totalTimetables"> <ul class="stats-list">
&bull; <li v-if="stats.totalTimetables">
<i18n-t keypath="journal.daily-stats.total"> <i18n-t keypath="journal.daily-stats.total">
<template #count> <template #count>
<b class="text--primary"> <b class="text--primary">
@@ -36,10 +35,9 @@
<b class="text--primary"> {{ stats.distanceSum?.toFixed(2) }} km</b> <b class="text--primary"> {{ stats.distanceSum?.toFixed(2) }} km</b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="stats.maxTimetable"> <li v-if="stats.maxTimetable">
&bull;
<i18n-t keypath="journal.daily-stats.longest"> <i18n-t keypath="journal.daily-stats.longest">
<template #id> <template #id>
<router-link :to="`/journal/timetables?search-train=%23${stats.maxTimetable.id}`"> <router-link :to="`/journal/timetables?search-train=%23${stats.maxTimetable.id}`">
@@ -60,10 +58,9 @@
<b class="text--primary">{{ stats.maxTimetable.routeDistance }} km</b> <b class="text--primary">{{ stats.maxTimetable.routeDistance }} km</b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="topDispatchers.length == 1"> <li v-if="topDispatchers.length == 1">
&bull;
<i18n-t keypath="journal.daily-stats.most-active-dr"> <i18n-t keypath="journal.daily-stats.most-active-dr">
<template #dispatcher> <template #dispatcher>
<router-link <router-link
@@ -79,10 +76,9 @@
</b> </b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="topDispatchers.length > 1"> <li v-if="topDispatchers.length > 1">
&bull;
<i18n-t keypath="journal.daily-stats.most-active-dr-many"> <i18n-t keypath="journal.daily-stats.most-active-dr-many">
<template #dispatchers> <template #dispatchers>
<span v-for="(disp, i) in topDispatchers" :key="i"> <span v-for="(disp, i) in topDispatchers" :key="i">
@@ -103,10 +99,9 @@
</b> </b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="stats.longestDuties.length > 0"> <li v-if="stats.longestDuties.length > 0">
&bull;
<i18n-t keypath="journal.daily-stats.longest-duties"> <i18n-t keypath="journal.daily-stats.longest-duties">
<template #dispatcher> <template #dispatcher>
<router-link <router-link
@@ -122,10 +117,9 @@
{{ calculateDuration(stats.longestDuties[0].duration) }} {{ calculateDuration(stats.longestDuties[0].duration) }}
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="stats.mostActiveDrivers.length > 0"> <li v-if="stats.mostActiveDrivers.length > 0">
&bull;
<i18n-t keypath="journal.daily-stats.most-active-driver"> <i18n-t keypath="journal.daily-stats.most-active-driver">
<template #driver> <template #driver>
<router-link <router-link
@@ -138,30 +132,30 @@
<b class="text--primary">{{ stats.mostActiveDrivers[0].distance.toFixed(2) }} km</b> <b class="text--primary">{{ stats.mostActiveDrivers[0].distance.toFixed(2) }} km</b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
</ul>
<hr class="section-separator" /> <hr class="section-separator" />
<div class="stats-badges"> <div class="stats-badges">
<span <span
class="stat-badge" class="badge stat-badge"
v-for="key in [ v-for="key in [
'rippedSwitches', 'rippedSwitches',
'derailments', 'derailments',
'skippedStopSignals', 'skippedStopSignals',
'radioStops', 'radioStops',
'kills' 'kills'
]" ]"
:key="key" :key="key"
> >
<span>{{ $t(`journal.daily-stats.${key}`) }}</span> <span>{{ $t(`journal.daily-stats.${key}`) }}</span>
<span>{{ <span>
Object.entries(stats.globalDiff).find(([k, v]) => k == key)?.[1] || '--' {{ Object.entries(stats.globalDiff).find(([k, v]) => k == key)?.[1] || '--' }}
}}</span>
</span> </span>
</div> </span>
</div> </div>
</span> </div>
</span> </span>
</section> </section>
</template> </template>
@@ -178,7 +172,6 @@ export default defineComponent({
name: 'journal-daily-stats', name: 'journal-daily-stats',
mixins: [dateMixin], mixins: [dateMixin],
// emits: ['toggleStatsOpen'],
data() { data() {
return { return {
@@ -193,7 +186,6 @@ export default defineComponent({
activated() { activated() {
this.startFetchingDailyStats(); this.startFetchingDailyStats();
// this.$emit('toggleStatsOpen', true);
}, },
deactivated() { deactivated() {
@@ -249,14 +241,24 @@ export default defineComponent({
.daily-stats { .daily-stats {
text-align: left; text-align: left;
} }
.daily-stats > span[data-active='0'] { .daily-stats > span[data-active='0'] {
opacity: 0.75; opacity: 0.75;
} }
ul.stats-list {
list-style: disc;
padding: 0 1em;
}
.stats-list a { .stats-list a {
text-decoration: underline; text-decoration: underline;
} }
.stats-list > li {
margin: 0.25em 0;
}
.stats-badges { .stats-badges {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
@@ -16,17 +16,17 @@
<hr class="header-separator" /> <hr class="header-separator" />
<div class="info-stats"> <div class="info-stats">
<span class="stat-badge" v-if="stats.services"> <span class="badge stat-badge" v-if="stats.services">
<span>{{ $t('journal.dispatcher-stats.services-count') }}</span> <span>{{ $t('journal.dispatcher-stats.services-count') }}</span>
<span>{{ stats.services.count }}</span> <span>{{ stats.services.count }}</span>
</span> </span>
<span class="stat-badge" v-if="stats.services"> <span class="badge stat-badge" v-if="stats.services">
<span>{{ $t('journal.dispatcher-stats.service-max') }}</span> <span>{{ $t('journal.dispatcher-stats.service-max') }}</span>
<span>{{ calculateDuration(stats.services.durationMax) }}</span> <span>{{ calculateDuration(stats.services.durationMax) }}</span>
</span> </span>
<span class="stat-badge" v-if="stats.services"> <span class="badge stat-badge" v-if="stats.services">
<span>{{ $t('journal.dispatcher-stats.service-avg') }}</span> <span>{{ $t('journal.dispatcher-stats.service-avg') }}</span>
<span>{{ calculateDuration(stats.services.durationAvg) }}</span> <span>{{ calculateDuration(stats.services.durationAvg) }}</span>
</span> </span>
@@ -35,22 +35,22 @@
<hr class="section-separator" /> <hr class="section-separator" />
<div class="info-stats"> <div class="info-stats">
<span class="stat-badge" v-if="stats.issuedTimetables"> <span class="badge stat-badge" v-if="stats.issuedTimetables">
<span>{{ $t('journal.dispatcher-stats.timetables-count') }}</span> <span>{{ $t('journal.dispatcher-stats.timetables-count') }}</span>
<span>{{ stats.issuedTimetables.count }}</span> <span>{{ stats.issuedTimetables.count }}</span>
</span> </span>
<span class="stat-badge" v-if="stats.issuedTimetables"> <span class="badge stat-badge" v-if="stats.issuedTimetables">
<span>{{ $t('journal.dispatcher-stats.timetables-sum') }}</span> <span>{{ $t('journal.dispatcher-stats.timetables-sum') }}</span>
<span>{{ stats.issuedTimetables.distanceSum.toFixed(2) }}km</span> <span>{{ stats.issuedTimetables.distanceSum.toFixed(2) }}km</span>
</span> </span>
<span class="stat-badge" v-if="stats.issuedTimetables"> <span class="badge stat-badge" v-if="stats.issuedTimetables">
<span>{{ $t('journal.dispatcher-stats.timetables-max') }}</span> <span>{{ $t('journal.dispatcher-stats.timetables-max') }}</span>
<span>{{ stats.issuedTimetables.distanceMax.toFixed(2) }}km</span> <span>{{ stats.issuedTimetables.distanceMax.toFixed(2) }}km</span>
</span> </span>
<span class="stat-badge" v-if="stats.issuedTimetables"> <span class="badge stat-badge" v-if="stats.issuedTimetables">
<span>{{ $t('journal.dispatcher-stats.timetables-avg') }}</span> <span>{{ $t('journal.dispatcher-stats.timetables-avg') }}</span>
<span>{{ stats.issuedTimetables.distanceAvg.toFixed(2) }}km</span> <span>{{ stats.issuedTimetables.distanceAvg.toFixed(2) }}km</span>
</span> </span>
@@ -1,48 +1,44 @@
<template> <template>
<transition name="status-anim" mode="out-in"> <div class="journal_warning" v-if="store.isOffline">
<div :key="dataStatus"> {{ $t('app.offline') }}
<div class="journal_warning" v-if="store.isOffline"> </div>
{{ $t('app.offline') }}
</div>
<Loading v-else-if="dataStatus == Status.Data.Loading" /> <Loading v-else-if="dataStatus == Status.Data.Loading" />
<div v-else-if="dataStatus == Status.Data.Error" class="journal_warning error"> <div v-else-if="dataStatus == Status.Data.Error" class="journal_warning error">
{{ $t('app.error') }} {{ $t('app.error') }}
</div> </div>
<div class="journal_warning" v-else-if="dispatcherHistory.length == 0"> <div class="journal_warning" v-else-if="dispatcherHistory.length == 0">
{{ $t('app.no-result') }} {{ $t('app.no-result') }}
</div> </div>
<ul v-else class="journal-list"> <div v-else>
<transition-group name="list-anim"> <transition-group name="list-anim" class="journal-list" tag="ul">
<JournalDispatcherEntry <JournalDispatcherEntry
v-for="entry in dispatcherHistory" v-for="entry in dispatcherHistory"
:key="entry.id" :key="entry.id"
:entry="entry" :entry="entry"
:onToggleShowExtraInfo="toggleExtraInfo" :onToggleShowExtraInfo="toggleExtraInfo"
:showExtraInfo="extraInfoIndexes.includes(entry.id)" :showExtraInfo="extraInfoIndexes.includes(entry.id)"
/> />
</transition-group> </transition-group>
<AddDataButton <AddDataButton
:list="dispatcherHistory" :list="dispatcherHistory"
:scrollDataLoaded="scrollDataLoaded" :scrollDataLoaded="scrollDataLoaded"
:scrollNoMoreData="scrollNoMoreData" :scrollNoMoreData="scrollNoMoreData"
@addHistoryData="addHistoryData" @addHistoryData="addHistoryData"
/> />
</ul> </div>
<div class="journal_warning" v-if="scrollNoMoreData"> <div class="journal_warning" v-if="scrollNoMoreData">
{{ $t('journal.no-further-data') }} {{ $t('journal.no-further-data') }}
</div> </div>
<div class="journal_warning" v-else-if="!scrollDataLoaded"> <div class="journal_warning" v-else-if="!scrollDataLoaded">
{{ $t('journal.loading-further-data') }} {{ $t('journal.loading-further-data') }}
</div> </div>
</div>
</transition>
</template> </template>
<script lang="ts"> <script lang="ts">
@@ -99,11 +95,4 @@ export default defineComponent({
<style lang="scss" scoped> <style lang="scss" scoped>
@import '../../../styles/variables.scss'; @import '../../../styles/variables.scss';
@import '../../../styles/JournalSection.scss'; @import '../../../styles/JournalSection.scss';
.journal-list {
display: flex;
flex-direction: column;
gap: 0.5em;
text-align: left;
}
</style> </style>
+10 -3
View File
@@ -30,7 +30,11 @@
</div> </div>
<transition name="dropdown-anim"> <transition name="dropdown-anim">
<div class="dropdown_wrapper" v-if="currentStatsTab !== null"> <div
class="dropdown_wrapper"
:class="{ 'dropdown-align-right': true }"
v-if="currentStatsTab !== null"
>
<keep-alive> <keep-alive>
<component :is="currentStatsTab" :key="currentStatsTab"></component> <component :is="currentStatsTab" :key="currentStatsTab"></component>
</keep-alive> </keep-alive>
@@ -79,7 +83,10 @@ export default defineComponent({
@import '../../styles/dropdown_filters.scss'; @import '../../styles/dropdown_filters.scss';
@import '../../styles/variables.scss'; @import '../../styles/variables.scss';
.dropdown_wrapper { .dropdown_wrapper.dropdown-align-right {
max-width: 100%; left: auto;
right: 0;
max-width: 700px;
// max-width: 100%;
} }
</style> </style>
@@ -1,29 +1,40 @@
<template> <template>
<div> <div>
<div class="details-actions"> <div class="details-actions">
<button class="btn--action"> <button class="btn--action" @click="toggleExtraInfo">
<b>{{ $t('journal.stock-info') }}</b> <b>{{ $t('journal.entry-details') }}</b>
<img :src="`/images/icon-arrow-${showExtraInfo ? 'asc' : 'desc'}.svg`" alt="Arrow icon" /> <img :src="`/images/icon-arrow-${showExtraInfo ? 'asc' : 'desc'}.svg`" alt="Arrow icon" />
</button> </button>
<router-link
v-if="driverRouteLocation !== null"
class="a-button btn--action btn-timetable"
:to="driverRouteLocation"
>
<img src="/images/icon-train.svg" alt="train icon" />
<b>{{ $t('journal.timetable-online-button') }}</b>
</router-link>
</div> </div>
<div class="details-body" v-if="timetable.stockString && timetable.stockMass && showExtraInfo"> <div class="details-body" v-if="showExtraInfo">
<hr /> <div class="g-separator"></div>
<EntryStops :timetable="timetable" />
<div class="g-separator"></div>
<div class="stock-specs"> <div class="stock-specs">
<span class="badge"> <span class="badge" v-if="timetable.authorName">
<span>{{ $t('journal.dispatcher-name') }}</span> <span>{{ $t('journal.dispatcher-name') }}</span>
<span>{{ timetable.authorName }}</span> <span>{{ timetable.authorName }}</span>
</span> </span>
</div>
<div class="stock-specs"> <span class="badge" v-if="timetable.maxSpeed">
<span class="badge">
<span>{{ $t('journal.stock-max-speed') }}</span> <span>{{ $t('journal.stock-max-speed') }}</span>
<span>{{ timetable.maxSpeed }}km/h</span> <span>{{ timetable.maxSpeed }}km/h</span>
</span> </span>
<span class="badge"> <span class="badge" v-if="timetable.stockLength">
<span>{{ $t('journal.stock-length') }}</span> <span>{{ $t('journal.stock-length') }}</span>
<span> <span>
{{ {{
@@ -34,13 +45,13 @@
</span> </span>
</span> </span>
<span class="badge"> <span class="badge" v-if="timetable.stockMass">
<span>{{ $t('journal.stock-mass') }}</span> <span>{{ $t('journal.stock-mass') }}</span>
<span> <span>
{{ {{
Math.floor( Math.floor(
(currentHistoryIndex == 0 (currentHistoryIndex == 0
? timetable.stockMass! ? timetable.stockMass
: stockHistory[currentHistoryIndex].stockMass || timetable.stockMass) / 1000 : stockHistory[currentHistoryIndex].stockMass || timetable.stockMass) / 1000
) )
}}t }}t
@@ -48,27 +59,56 @@
</span> </span>
</div> </div>
<!-- Historia zmian w składzie --> <div class="stock-dangers" v-if="timetable.twr || timetable.skr">
<div class="stock-history" v-if="stockHistory.length > 1"> <div class="g-separator"></div>
<button
v-for="(sh, i) in stockHistory" <b>{{ $t('journal.stock-dangers') }}:</b>
:key="i"
class="btn--action" <ul>
:data-checked="i == currentHistoryIndex" <li v-if="timetable.twr">
@click.stop="currentHistoryIndex = i" <b class="text--primary">{{ $t('general.TWR') }} (TWR)</b>
> <span v-if="timetable.warningNotes">
{{ sh.updatedAt }} | <i>{{ timetable.warningNotes }}</i>
</button> </span>
</li>
<li v-if="timetable.skr">
<b class="text--primary">{{ $t('general.SKR') }}</b>
<span v-if="timetable.warningNotes">
| Komentarze: <i>{{ timetable.warningNotes }}</i>
</span>
</li>
</ul>
</div> </div>
<StockList <!-- Historia zmian w składzie -->
:trainStockList=" <div v-if="timetable.stockString || stockHistory.length != 0">
(currentHistoryIndex == 0 <div class="g-separator"></div>
? timetable.stockString <b>{{ $t('journal.stock-preview') }}:</b>
: stockHistory[currentHistoryIndex].stockString
).split(';') <div class="stock-history" v-if="stockHistory.length > 1">
" <button
/> v-for="(sh, i) in stockHistory"
:key="i"
class="btn--action"
:data-checked="i == currentHistoryIndex"
@click.stop="currentHistoryIndex = i"
>
{{ sh.updatedAt }}
</button>
</div>
<div v-if="timetable.stockString" style="margin-top: 1em">
<StockList
:trainStockList="
(currentHistoryIndex == 0
? timetable.stockString
: stockHistory[currentHistoryIndex].stockString
).split(';')
"
/>
</div>
</div>
</div> </div>
</div> </div>
</template> </template>
@@ -77,9 +117,14 @@
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import StockList from '../../Global/StockList.vue'; import StockList from '../../Global/StockList.vue';
import { API } from '../../../typings/api'; import { API } from '../../../typings/api';
import { RouteLocationRaw } from 'vue-router';
import EntryStops from './EntryStops.vue';
export default defineComponent({ export default defineComponent({
components: { StockList }, components: { StockList, EntryStops },
emits: ['toggleExtraInfo'],
props: { props: {
showExtraInfo: { showExtraInfo: {
type: Boolean, type: Boolean,
@@ -112,12 +157,25 @@ export default defineComponent({
stockLength: Number(historyData[3]) || undefined stockLength: Number(historyData[3]) || undefined
}; };
}); });
},
driverRouteLocation(): RouteLocationRaw | null {
if (this.timetable.terminated) return null;
return {
name: 'DriverView',
query: {
trainId: `${this.timetable.driverId}|${this.timetable.trainNo}|eu`
}
};
} }
}, },
methods: { methods: {
onImageError(e: Event) { onImageError(e: Event) {
const imageEl = e.target as HTMLImageElement; const imageEl = e.target as HTMLImageElement;
imageEl.src = '/images/icon-unknown.png'; imageEl.src = '/images/icon-unknown.png';
},
toggleExtraInfo() {
this.$emit('toggleExtraInfo', this.timetable.id);
} }
} }
}); });
@@ -134,6 +192,8 @@ export default defineComponent({
.details-actions { .details-actions {
display: flex; display: flex;
gap: 0.5em;
margin-top: 1em;
button img { button img {
height: 1.25em; height: 1.25em;
@@ -155,7 +215,6 @@ export default defineComponent({
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
gap: 0.5em; gap: 0.5em;
margin-top: 0.5em;
.badge { .badge {
margin: 0; margin: 0;
@@ -167,20 +226,14 @@ export default defineComponent({
} }
} }
ul.stock-list { hr {
display: flex; margin: 0.5em 0;
align-items: flex-end; }
overflow: auto;
padding-bottom: 0.5em; .stock-dangers ul {
list-style: disc;
li > div { padding-left: 1em;
margin: 1em 0; padding-top: 0.5em;
text-align: center;
color: #aaa;
font-size: 0.9em;
}
} }
@include smallScreen() { @include smallScreen() {
@@ -3,9 +3,23 @@
<span class="general-train"> <span class="general-train">
<span class="text--grayed">#{{ timetable.id }}</span> <span class="text--grayed">#{{ timetable.id }}</span>
<span class="badges" v-if="timetable.skr || timetable.twr"> <span
<span class="train-badge twr" v-if="timetable.twr" :title="$t('general.TWR')">TWR</span> class="train-badge twr"
<span class="train-badge skr" v-if="timetable.skr" :title="$t('general.SKR')">SKR</span> v-if="timetable.twr"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="
$t('general.TWR') + `${timetable.warningNotes ? ':\n' + timetable.warningNotes : ''}`
"
>
TWR
</span>
<span
class="train-badge skr"
v-if="timetable.skr"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('general.SKR')"
>
SKR
</span> </span>
<span> <span>
@@ -27,18 +41,19 @@
{{ timetable.driverLevel < 2 ? 'L' : `${timetable.driverLevel}` }} {{ timetable.driverLevel < 2 ? 'L' : `${timetable.driverLevel}` }}
</strong> </strong>
<strong <router-link
v-if="apiStore.donatorsData.includes(timetable.driverName)" v-if="apiStore.donatorsData.includes(timetable.driverName)"
class="text--donator" class="text--donator"
data-tooltip-type="DonatorTooltip" data-tooltip-type="DonatorTooltip"
:data-tooltip-content="$t('donations.driver-message')" :data-tooltip-content="$t('donations.driver-message')"
:to="`/journal/timetables?search-driver=${timetable.driverName}`"
> >
{{ timetable.driverName }} <strong>{{ timetable.driverName }}</strong>
</strong> </router-link>
<strong v-else> <router-link v-else :to="`/journal/timetables?search-driver=${timetable.driverName}`">
{{ timetable.driverName }} <strong>{{ timetable.driverName }}</strong>
</strong> </router-link>
</span> </span>
<span class="general-time"> <span class="general-time">
@@ -66,15 +81,6 @@
: `${$t('journal.timetable-abandoned')} ${localeTime(timetable.endDate, $i18n.locale)}` : `${$t('journal.timetable-abandoned')} ${localeTime(timetable.endDate, $i18n.locale)}`
}} }}
</b> </b>
<button
v-if="timetable.terminated == false"
class="btn--action btn-timetable"
@click.stop="showTimetable(timetable, $event.currentTarget)"
>
<img src="/images/icon-train.svg" alt="train icon" />
<b>{{ $t('journal.timetable-online-button') }}</b>
</button>
</span> </span>
</div> </div>
</template> </template>
@@ -84,13 +90,12 @@ import { PropType, defineComponent } from 'vue';
import { API } from '../../../typings/api'; import { API } from '../../../typings/api';
import dateMixin from '../../../mixins/dateMixin'; import dateMixin from '../../../mixins/dateMixin';
import modalTrainMixin from '../../../mixins/modalTrainMixin';
import styleMixin from '../../../mixins/styleMixin'; import styleMixin from '../../../mixins/styleMixin';
import { useApiStore } from '../../../store/apiStore'; import { useApiStore } from '../../../store/apiStore';
import trainCategoryMixin from '../../../mixins/trainCategoryMixin'; import trainCategoryMixin from '../../../mixins/trainCategoryMixin';
export default defineComponent({ export default defineComponent({
mixins: [dateMixin, modalTrainMixin, styleMixin, trainCategoryMixin], mixins: [dateMixin, styleMixin, trainCategoryMixin],
data() { data() {
return { return {
@@ -103,14 +108,6 @@ export default defineComponent({
type: Object as PropType<API.TimetableHistory.Data>, type: Object as PropType<API.TimetableHistory.Data>,
required: true required: true
} }
},
methods: {
showTimetable(timetable: API.TimetableHistory.Data, target: EventTarget | null) {
if (timetable?.terminated) return;
this.selectModalTrainById(`${timetable.driverName}${timetable.trainNo}`, target);
}
} }
}); });
</script> </script>
@@ -137,7 +134,6 @@ export default defineComponent({
gap: 0.25em; gap: 0.25em;
cursor: pointer; cursor: pointer;
line-height: 2;
} }
.general-time { .general-time {
@@ -180,6 +176,7 @@ export default defineComponent({
@include smallScreen { @include smallScreen {
.item-general { .item-general {
flex-direction: column;
justify-content: center; justify-content: center;
} }
} }
@@ -1,5 +1,5 @@
<template> <template>
<div class="item-status" style="margin: 0.5em 0"> <div class="entry-status" style="margin: 0.5em 0">
<ProgressBar <ProgressBar
:progressPercent="~~((timetable.currentDistance / timetable.routeDistance) * 100)" :progressPercent="~~((timetable.currentDistance / timetable.routeDistance) * 100)"
:progressType="!timetable.fulfilled && timetable.terminated ? 'abandoned' : ''" :progressType="!timetable.fulfilled && timetable.terminated ? 'abandoned' : ''"
@@ -61,7 +61,7 @@ export default defineComponent({
<style lang="scss" scoped> <style lang="scss" scoped>
@import '../../../styles/responsive.scss'; @import '../../../styles/responsive.scss';
.item-status { .entry-status {
display: flex; display: flex;
align-items: center; align-items: center;
flex-wrap: wrap; flex-wrap: wrap;
@@ -0,0 +1,296 @@
<template>
<div class="entry-stops">
<ul class="stop-list">
<li v-for="(stop, i) in timetableStops" :key="stop.stopName">
<span class="stop-label" :data-confirmed="stop.isConfirmed">
<span v-if="i > 0">&gt;</span>
<span class="stop-name">{{ stop.stopName }}</span>
<span
class="stop-date"
v-if="stop.scheduledArrivalTimestamp != 0"
:data-delayed="
stop.isConfirmed && stop.arrivalTimestamp - stop.scheduledArrivalTimestamp > 0
"
:data-preponed="
stop.isConfirmed &&
stop.arrivalTimestamp != 0 &&
stop.arrivalTimestamp - stop.scheduledArrivalTimestamp < 0
"
>
<span
v-if="stop.isConfirmed && stop.arrivalTimestamp - stop.scheduledArrivalTimestamp != 0"
>
p. <s>{{ timestampToString(stop.scheduledArrivalTimestamp) }}</s>
{{ timestampToString(stop.arrivalTimestamp) }}
</span>
<span v-else>p. {{ timestampToString(stop.scheduledArrivalTimestamp) }}</span>
</span>
<span
class="stop-time"
v-if="stop.stopTime > 0"
:data-stop-ph="stop.stopType.includes('ph')"
:data-stop-pt="stop.stopType.includes('pt')"
:data-stop-pm="stop.stopType.includes('pm')"
>
/<span>{{ stop.stopTime }} {{ stop.stopType }}</span
>/
</span>
<span
class="stop-date"
v-if="
stop.scheduledDepartureTimestamp != 0 &&
stop.scheduledArrivalTimestamp != stop.scheduledDepartureTimestamp
"
:data-delayed="
stop.isConfirmed && stop.departureTimestamp - stop.scheduledDepartureTimestamp > 0
"
:data-preponed="
stop.isConfirmed &&
stop.departureTimestamp != 0 &&
stop.departureTimestamp - stop.scheduledDepartureTimestamp < 0
"
>
<span
v-if="
stop.isConfirmed && stop.departureTimestamp - stop.scheduledDepartureTimestamp != 0
"
>
o. <s>{{ timestampToString(stop.scheduledDepartureTimestamp) }}</s>
{{ timestampToString(stop.departureTimestamp) }}
</span>
<span v-else>o. {{ timestampToString(stop.scheduledDepartureTimestamp) }}</span>
</span>
</span>
</li>
</ul>
<ul class="timetable-path-list" v-if="timetablePathDetails">
<li
v-for="(pathData, i) in timetablePathDetails"
:data-visited="pathData.isVisited"
:data-next-visited="
i < timetablePathDetails.length - 1 && timetablePathDetails[i + 1].isVisited
"
>
<span v-if="i > 0" class="path-arrow">&gt;</span>
<span class="path-arrival" v-if="pathData.arrival">{{ pathData.arrival }}</span>
<b class="path-scenery">{{ pathData.sceneryName }}</b>
<span class="path-departure" v-if="pathData.departure">{{ pathData.departure }}</span>
</li>
</ul>
</div>
</template>
<script lang="ts">
import { PropType, defineComponent } from 'vue';
import dateMixin from '../../../mixins/dateMixin';
import { API } from '../../../typings/api';
interface ITimetableStopDetails {
stopName: string;
arrivalTimestamp: number;
scheduledArrivalTimestamp: number;
departureTimestamp: number;
scheduledDepartureTimestamp: number;
stopTime: number;
stopType: string;
isConfirmed: boolean;
}
export default defineComponent({
mixins: [dateMixin],
props: {
timetable: {
type: Object as PropType<API.TimetableHistory.Data>,
required: true
}
},
computed: {
timetablePathDetails() {
if (!this.timetable.path || this.timetable.path == '') return null;
return this.timetable.path.split(';').map((pathEl, i) => {
const [arrival, name, departure] = pathEl.split(',');
const sceneryName = name.split(' ').slice(0, -1).join(' ');
const sceneryHash = name.split(' ').pop()?.replace('.sc', '') ?? '';
const isVisited = this.timetable.visitedSceneries.includes(sceneryHash);
return {
arrival,
sceneryName,
sceneryHash,
departure,
isVisited,
isVisitedOffline:
!isVisited &&
this.timetable.visitedSceneries.includes(`${sceneryName} ${sceneryHash}.sc`)
};
});
},
timetableStops(): ITimetableStopDetails[] {
const timetable = this.timetable;
const stopNames = timetable.sceneriesString.split('%');
return stopNames.reduce<ITimetableStopDetails[]>((acc, stopName, i, arr) => {
const arrivalDate =
i == arr.length - 1
? (timetable.checkpointArrivals.at(i) ?? timetable.endDate)
: timetable.checkpointArrivals.at(i);
const scheduledArrivalDate =
i == arr.length - 1
? (timetable.checkpointArrivalsScheduled.at(i) ?? timetable.scheduledEndDate)
: timetable.checkpointArrivalsScheduled.at(i);
const departureDate =
i == 0
? (timetable.checkpointDepartures.at(i) ?? timetable.beginDate)
: timetable.checkpointDepartures.at(i);
const scheduledDepartureDate =
i == 0
? (timetable.checkpointDeparturesScheduled.at(i) ?? timetable.scheduledBeginDate)
: timetable.checkpointDeparturesScheduled.at(i);
const stopTime = Number(timetable.checkpointStopTypes.at(i)?.split(',')[0]) || 0;
const stopType = timetable.checkpointStopTypes.at(i)?.split(',').slice(1).join(',') || 'pt';
acc.push({
stopName,
arrivalTimestamp: this.dateStringToTimestamp(arrivalDate),
scheduledArrivalTimestamp: this.dateStringToTimestamp(scheduledArrivalDate),
departureTimestamp: this.dateStringToTimestamp(departureDate),
scheduledDepartureTimestamp: this.dateStringToTimestamp(scheduledDepartureDate),
stopTime,
stopType,
isConfirmed: i < timetable.confirmedStopsCount
});
return acc;
}, []);
}
}
});
</script>
<style lang="scss" scoped>
@import '../../../styles/badge.scss';
.entry-stops {
word-wrap: break-word;
gap: 0.25em;
font-size: 0.95em;
}
.stop-list {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
padding: 0.5em 0;
}
.stop-label {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
align-items: center;
color: white;
&[data-confirmed='true'] > .stop-name {
color: lightgreen;
}
&[data-confirmed='true'] > .stop-date:not([data-preponed='true']):not([data-delayed='true']) {
color: lightgreen;
}
}
.stop-name {
font-weight: bold;
color: #ccc;
}
.stop-date {
color: #ccc;
s {
color: #aaa;
}
&[data-delayed='true'] {
color: salmon;
}
&[data-preponed='true'] {
color: mediumspringgreen;
}
}
.stop-time {
&[data-stop-pt='true'] span {
color: #999;
}
&[data-stop-ph='true'] span,
&[data-stop-pm='true'] span {
color: gold;
}
}
.timetable-path-list {
display: flex;
flex-wrap: wrap;
gap: 0.5em 0;
padding: 0.5em 0;
color: #ccc;
li > .path-scenery:first-child,
li > .path-arrival:nth-child(2) {
border-radius: 0.5em 0 0 0.5em;
}
li > :last-child {
border-radius: 0 0.5em 0.5em 0;
}
}
.path-scenery {
padding: 0.25em 0.5em;
background-color: #303030;
}
.path-arrival,
.path-departure {
padding: 0.25em;
display: inline-block;
background-color: #4e4e4e;
min-width: 25px;
text-align: center;
}
.path-arrow {
padding: 0 0.5em;
}
.timetable-path-list > li[data-visited='true'] {
.path-arrival,
.path-scenery,
.path-arrow {
color: lightgreen;
}
&[data-next-visited='true'] .path-departure {
color: lightgreen;
}
}
</style>
@@ -12,7 +12,7 @@
<hr class="header-separator" /> <hr class="header-separator" />
<div class="info-stats"> <div class="info-stats">
<span class="stat-badge"> <span class="badge stat-badge">
<span>{{ $t('journal.driver-stats.timetables') }}</span> <span>{{ $t('journal.driver-stats.timetables') }}</span>
<span <span
>{{ store.driverStatsData._count.fulfilled }} / >{{ store.driverStatsData._count.fulfilled }} /
@@ -20,17 +20,17 @@
> >
</span> </span>
<span class="stat-badge"> <span class="badge stat-badge">
<span>{{ $t('journal.driver-stats.longest-timetable') }}</span> <span>{{ $t('journal.driver-stats.longest-timetable') }}</span>
<span> {{ store.driverStatsData._max.routeDistance.toFixed(2) }}km </span> <span> {{ store.driverStatsData._max.routeDistance.toFixed(2) }}km </span>
</span> </span>
<span class="stat-badge"> <span class="badge stat-badge">
<span>{{ $t('journal.driver-stats.avg-timetable') }}</span> <span>{{ $t('journal.driver-stats.avg-timetable') }}</span>
<span> {{ store.driverStatsData._avg.routeDistance.toFixed(2) }}km </span> <span> {{ store.driverStatsData._avg.routeDistance.toFixed(2) }}km </span>
</span> </span>
<span class="stat-badge"> <span class="badge stat-badge">
<span>{{ $t('journal.driver-stats.distance') }}</span> <span>{{ $t('journal.driver-stats.distance') }}</span>
<span> <span>
{{ store.driverStatsData._sum.currentDistance.toFixed(2) }} / {{ store.driverStatsData._sum.currentDistance.toFixed(2) }} /
@@ -38,7 +38,7 @@
</span> </span>
</span> </span>
<span class="stat-badge"> <span class="badge stat-badge">
<span>{{ $t('journal.driver-stats.stations') }}</span> <span>{{ $t('journal.driver-stats.stations') }}</span>
<span> <span>
{{ store.driverStatsData._sum.confirmedStopsCount }} / {{ store.driverStatsData._sum.confirmedStopsCount }} /
@@ -0,0 +1,149 @@
<template>
<li class="timetable-history-entry">
<!-- General -->
<EntryGeneral :timetable="timetableEntry" />
<div @click="toggleExtraInfo" style="cursor: pointer">
<!-- Route -->
<div class="entry-route">
<b>{{ timetableEntry.route.replace('|', ' - ') }}</b>
</div>
<hr />
<!-- Status -->
<EntryStatus :timetable="timetableEntry" />
</div>
<!-- Extra -->
<EntryDetails
:timetable="timetableEntry"
:show-extra-info="showExtraInfo"
@toggle-extra-info="toggleExtraInfo"
/>
</li>
</template>
<script lang="ts">
import { defineComponent, PropType } from 'vue';
import { API } from '../../../typings/api';
import { useApiStore } from '../../../store/apiStore';
import { Journal } from '../typings';
import trainCategoryMixin from '../../../mixins/trainCategoryMixin';
import dateMixin from '../../../mixins/dateMixin';
import styleMixin from '../../../mixins/styleMixin';
import EntryGeneral from './EntryGeneral.vue';
import EntryStatus from './EntryStatus.vue';
import EntryDetails from './EntryDetails.vue';
export default defineComponent({
props: {
timetableEntry: {
type: Object as PropType<API.TimetableHistory.Data>,
required: true
},
showExtraInfo: {
type: Boolean,
required: true
}
},
components: { EntryDetails, EntryGeneral, EntryStatus },
mixins: [trainCategoryMixin, dateMixin, styleMixin],
emits: ['toggleShowExtraInfo'],
data() {
return {
apiStore: useApiStore()
};
},
computed: {
timetablePathDetails() {
if (!this.timetableEntry.path || this.timetableEntry.path == '') return null;
return this.timetableEntry.path.split(';').map((pathEl, i) => {
const [arrival, name, departure] = pathEl.split(',');
const sceneryName = name.split(' ').slice(0, -1).join(' ');
const sceneryHash = name.split(' ').pop()?.replace('.sc', '') ?? '';
return {
arrival,
sceneryName,
sceneryHash,
departure,
isVisited: this.timetableEntry.visitedSceneries?.includes(sceneryHash) ?? false
};
});
},
timetableStops(): Journal.TimetableStopDetails[] {
const timetableEntry = this.timetableEntry;
const stopNames = timetableEntry.sceneriesString.split('%');
return stopNames.reduce<Journal.TimetableStopDetails[]>((acc, stopName, i, arr) => {
const arrivalDate =
i == arr.length - 1
? (timetableEntry.checkpointArrivals.at(i) ?? timetableEntry.endDate)
: timetableEntry.checkpointArrivals.at(i);
const scheduledArrivalDate =
i == arr.length - 1
? (timetableEntry.checkpointArrivalsScheduled.at(i) ?? timetableEntry.scheduledEndDate)
: timetableEntry.checkpointArrivalsScheduled.at(i);
const departureDate =
i == 0
? (timetableEntry.checkpointDepartures.at(i) ?? timetableEntry.beginDate)
: timetableEntry.checkpointDepartures.at(i);
const scheduledDepartureDate =
i == 0
? (timetableEntry.checkpointDeparturesScheduled.at(i) ??
timetableEntry.scheduledBeginDate)
: timetableEntry.checkpointDeparturesScheduled.at(i);
const stopTime = Number(timetableEntry.checkpointStopTypes.at(i)?.split(',')[0]) || 0;
const stopType = timetableEntry.checkpointStopTypes.at(i)?.split(',')[1] || '';
acc.push({
stopName,
arrivalTimestamp: this.dateStringToTimestamp(arrivalDate),
scheduledArrivalTimestamp: this.dateStringToTimestamp(scheduledArrivalDate),
departureTimestamp: this.dateStringToTimestamp(departureDate),
scheduledDepartureTimestamp: this.dateStringToTimestamp(scheduledDepartureDate),
stopTime,
stopType,
isConfirmed: i < timetableEntry.confirmedStopsCount
});
return acc;
}, []);
}
},
methods: {
toggleExtraInfo() {
this.$emit('toggleShowExtraInfo');
}
}
});
</script>
<style lang="scss" scoped>
@import '../../../styles/responsive.scss';
.timetable-history-entry {
background-color: #1a1a1a;
padding: 1em;
}
@include smallScreen {
.entry-route {
text-align: center;
}
}
</style>
@@ -1,62 +1,40 @@
<template> <template>
<div> <div>
<transition name="status-anim" mode="out-in"> <div class="journal_warning" v-if="store.isOffline">
<div :key="dataStatus"> {{ $t('app.offline') }}
<div class="journal_warning" v-if="store.isOffline"> </div>
{{ $t('app.offline') }}
</div>
<Loading v-else-if="dataStatus == Status.Data.Loading" /> <Loading v-else-if="dataStatus == Status.Data.Loading" />
<div v-else-if="dataStatus == Status.Data.Error" class="journal_warning error"> <div v-else-if="dataStatus == Status.Data.Error" class="journal_warning error">
{{ $t('app.error') }} {{ $t('app.error') }}
</div> </div>
<div v-else-if="timetableHistory.length == 0" class="journal_warning"> <div v-else-if="timetableHistory.length == 0" class="journal_warning">
{{ $t('app.no-result') }} {{ $t('app.no-result') }}
</div> </div>
<div v-else> <div v-else>
<ul class="journal-list"> <transition-group name="list-anim" class="journal-list" tag="ul">
<transition-group name="list-anim"> <JournalTimetableEntry
<li v-for="(timetableEntry, i) in timetableHistory"
v-for="{ timetable, showExtraInfo } in computedTimetableHistory" :key="timetableEntry.id"
class="journal_item" :timetableEntry="timetableEntry"
:key="timetable.id" :onToggleShowExtraInfo="() => toggleExtraInfo(timetableEntry.id)"
@click="showExtraInfo.value = !showExtraInfo.value" :showExtraInfo="extraInfoIndexes.includes(timetableEntry.id)"
> />
<div class="journal_item-info"> </transition-group>
<!-- General -->
<TimetableGeneral :timetable="timetable" />
<!-- Route -->
<span class="item-route">
<b>{{ timetable.route.replace('|', ' - ') }}</b>
</span>
<hr /> <AddDataButton
<!-- Stops --> :list="timetableHistory"
<TimetableStops :timetable="timetable" :showExtraInfo="showExtraInfo.value" /> :scrollDataLoaded="scrollDataLoaded"
<!-- Status --> :scrollNoMoreData="scrollNoMoreData"
<TimetableStatus :timetable="timetable" /> @addHistoryData="addHistoryData"
/>
<!-- Extra --> </div>
<TimetableDetails :timetable="timetable" :showExtraInfo="showExtraInfo.value" />
</div>
</li>
</transition-group>
</ul>
<AddDataButton
:list="timetableHistory"
:scrollDataLoaded="scrollDataLoaded"
:scrollNoMoreData="scrollNoMoreData"
@addHistoryData="addHistoryData"
/>
</div>
</div>
</transition>
<div class="journal_warning" v-if="scrollNoMoreData">{{ $t('journal.no-further-data') }}</div> <div class="journal_warning" v-if="scrollNoMoreData">{{ $t('journal.no-further-data') }}</div>
<div class="journal_warning" v-else-if="!scrollDataLoaded"> <div class="journal_warning" v-else-if="!scrollDataLoaded">
{{ $t('journal.loading-further-data') }} {{ $t('journal.loading-further-data') }}
</div> </div>
@@ -64,28 +42,21 @@
</template> </template>
<script lang="ts"> <script lang="ts">
import { defineComponent, PropType, ref } from 'vue'; import { defineComponent, PropType } from 'vue';
import Loading from '../../Global/Loading.vue'; import Loading from '../../Global/Loading.vue';
import AddDataButton from '../../Global/AddDataButton.vue'; import AddDataButton from '../../Global/AddDataButton.vue';
import JournalTimetableEntry from './JournalTimetableEntry.vue';
import { useMainStore } from '../../../store/mainStore'; import { useMainStore } from '../../../store/mainStore';
import { Status } from '../../../typings/common'; import { Status } from '../../../typings/common';
import { API } from '../../../typings/api'; import { API } from '../../../typings/api';
import TimetableGeneral from './TimetableGeneral.vue';
import TimetableStops from './TimetableStops.vue';
import TimetableStatus from './TimetableStatus.vue';
import TimetableDetails from './TimetableDetails.vue';
export default defineComponent({ export default defineComponent({
components: { components: {
Loading, Loading,
AddDataButton, AddDataButton,
TimetableDetails, JournalTimetableEntry
TimetableGeneral,
TimetableStatus,
TimetableStops
}, },
props: { props: {
@@ -110,16 +81,29 @@ export default defineComponent({
data() { data() {
return { return {
Status, Status,
store: useMainStore() store: useMainStore(),
extraInfoIndexes: [] as number[]
}; };
}, },
computed: { watch: {
computedTimetableHistory() { '$route.query': {
return this.timetableHistory.map((timetable) => ({ deep: true,
timetable, handler() {
showExtraInfo: ref(false) this.extraInfoIndexes.length = 0;
}));
this.$nextTick(() => {
console.log(this.$el.querySelector('ul'));
});
}
}
},
methods: {
toggleExtraInfo(id: number) {
const existingIdx = this.extraInfoIndexes.indexOf(id);
if (existingIdx != -1) this.extraInfoIndexes.splice(existingIdx, 1);
else this.extraInfoIndexes.push(id);
} }
} }
}); });
@@ -1,164 +0,0 @@
<template>
<div class="timetable-stops">
<div class="stop-list">
<span
v-for="(stop, i) in timetableStops.filter((_, i) =>
!showExtraInfo ? i == 0 || i == timetableStops.length - 1 : true
)"
class="stop-list-item"
:key="stop.stopName"
:data-confirmed="stop.confirmed"
>
<span v-if="i > 0">
&gt;
<span v-if="!showExtraInfo && i == 1 && timetableStops.length > 2">
... (+{{ timetableStops.length - 2 }}) &gt;
</span>
</span>
<span class="stop-name">{{ stop.stopName }}</span>
<span v-html="stop.html"></span>
</span>
</div>
<div class="path-details" v-if="showExtraInfo && timetablePathDetails">
<span
v-for="(pathData, i) in timetablePathDetails"
:data-visited="pathData.isVisited"
:data-next-visited="
i < timetablePathDetails.length - 1 && timetablePathDetails[i + 1].isVisited
"
>
<span class="path-arrival" v-if="pathData.arrival">/ {{ pathData.arrival }} &RightArrow; </span>
<b class="path-scenery">{{ pathData.sceneryName }}</b>
<span class="path-departure" v-if="pathData.departure">
&RightArrow; {{ pathData.departure }}&nbsp;
</span>
</span>
</div>
</div>
</template>
<script lang="ts">
import { PropType, defineComponent } from 'vue';
import dateMixin from '../../../mixins/dateMixin';
import { API } from '../../../typings/api';
export default defineComponent({
mixins: [dateMixin],
props: {
showExtraInfo: {
type: Boolean,
required: true
},
timetable: {
type: Object as PropType<API.TimetableHistory.Data>,
required: true
}
},
computed: {
timetablePathDetails() {
if (!this.timetable.path || this.timetable.path == '') return null;
return this.timetable.path.split(';').map((pathEl, i) => {
const [arrival, name, departure] = pathEl.split(',');
const sceneryName = name.split(' ').slice(0, -1).join(' ');
const sceneryHash = name.split(' ').pop()?.replace('.sc', '') ?? '';
return {
arrival,
sceneryName,
sceneryHash,
departure,
isVisited: this.timetable.visitedSceneries?.includes(sceneryHash) ?? false
};
});
},
timetableStops() {
const timetable = this.timetable;
const stopNames = timetable.sceneriesString.split('%');
const beginDateHTML = ` (o. ${
timetable.beginDate != timetable.scheduledBeginDate
? `<s class="text--grayed">${this.localeTime(timetable.beginDate, this.$i18n.locale)}</s>`
: ''
} <span>${this.localeTime(timetable.scheduledBeginDate, this.$i18n.locale)}</span>)`;
const endDateHTML = ` (p. ${
timetable.endDate != timetable.scheduledEndDate && timetable.fulfilled
? `<s class="text--grayed">${this.localeTime(timetable.endDate, this.$i18n.locale)}</s>`
: ''
} <span>${this.localeTime(timetable.scheduledEndDate, this.$i18n.locale)}</span>)`;
return stopNames.map((stopName, i) => {
const confirmed = i < timetable.confirmedStopsCount;
if (i == 0) return { stopName, html: beginDateHTML, confirmed };
if (i == stopNames.length - 1) return { stopName, html: endDateHTML, confirmed };
const departureDateScheduled = this.stringToDate(
timetable.checkpointDeparturesScheduled?.at(i)
);
const departureDateReal = this.stringToDate(timetable.checkpointDepartures?.at(i));
const arrivalDateScheduled = this.stringToDate(
timetable.checkpointArrivalsScheduled?.at(i)
);
const arrivalDateReal = this.stringToDate(timetable.checkpointArrivals?.at(i));
const arrivalHTML =
(arrivalDateReal &&
arrivalDateScheduled &&
arrivalDateReal?.getTime() != arrivalDateScheduled?.getTime()
? `<s class="text--grayed">${this.parseDateToTimeString(arrivalDateScheduled)}</s> `
: '') + this.parseDateToTimeString(arrivalDateReal || arrivalDateScheduled);
const departureHTML =
(departureDateReal &&
departureDateScheduled &&
departureDateReal?.getTime() != departureDateScheduled?.getTime()
? `<s class="text--grayed">${this.parseDateToTimeString(departureDateScheduled)}</s> `
: '') + this.parseDateToTimeString(departureDateReal || departureDateScheduled);
let html = `${arrivalHTML}${departureHTML ? ` / ${departureHTML}` : ''}`;
if (html) html = ` (${html})`;
return { stopName, html, confirmed };
});
}
}
});
</script>
<style lang="scss" scoped>
.timetable-stops {
word-wrap: break-word;
gap: 0.25em;
font-size: 0.95em;
color: #adadad;
}
.stop-list {
&-item[data-confirmed='true'] {
color: lightgreen;
.stop-name {
font-weight: bold;
}
}
}
.path-details {
margin-top: 0.5em;
}
.path-details > span[data-visited='true'] {
.path-arrival,
.path-scenery {
color: lightgreen;
}
&[data-next-visited='true'] .path-departure {
color: lightgreen;
}
}
</style>
+12
View File
@@ -66,4 +66,16 @@ export namespace Journal {
iconName: string; iconName: string;
disabled: boolean; disabled: boolean;
} }
export interface TimetableStopDetails {
stopName: string;
arrivalTimestamp: number;
scheduledArrivalTimestamp: number;
departureTimestamp: number;
scheduledDepartureTimestamp: number;
stopTime: number;
stopType: string;
isConfirmed: boolean;
}
} }
@@ -15,14 +15,13 @@
<li <li
v-for="{ train, status } in stationTrains" v-for="{ train, status } in stationTrains"
class="badge user" class="badge user"
tabindex="0"
:key="train.id" :key="train.id"
:data-status="status" :data-status="status"
@click.prevent="selectModalTrain(train, $event.currentTarget)"
@keydown.enter="selectModalTrain(train, $event.currentTarget)"
> >
<span class="user_train">{{ train.trainNo }}</span> <router-link :to="train.driverRouteLocation" class="a-block">
<span class="user_name">{{ train.driverName }}</span> <span class="user_train">{{ train.trainNo }}</span>
<span class="user_name">{{ train.driverName }}</span>
</router-link>
</li> </li>
</transition-group> </transition-group>
</section> </section>
@@ -30,14 +29,13 @@
<script lang="ts"> <script lang="ts">
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import modalTrainMixin from '../../../mixins/modalTrainMixin';
import routerMixin from '../../../mixins/routerMixin'; import routerMixin from '../../../mixins/routerMixin';
import { ActiveScenery, Station, StopStatus } from '../../../typings/common'; import { ActiveScenery, Station } from '../../../typings/common';
import { getTrainStopStatus } from '../utils'; import { getTrainStopStatus } from '../utils';
import { useMainStore } from '../../../store/mainStore'; import { useMainStore } from '../../../store/mainStore';
export default defineComponent({ export default defineComponent({
mixins: [routerMixin, modalTrainMixin], mixins: [routerMixin],
props: { props: {
onlineScenery: { onlineScenery: {
@@ -99,38 +97,27 @@ ul {
} }
.user { .user {
cursor: pointer;
&_train {
color: black;
background-color: $no-timetable;
transition: background-color 200ms;
-ms-transition: background-color 200ms;
-webkit-transition: background-color 200ms;
}
&[data-status='no-timetable'] .user_train { &[data-status='no-timetable'] .user_train {
background-color: $no-timetable; background-color: $no-timetable;
} }
&[data-status='departed'] > &_train { &[data-status='departed'] .user_train {
background-color: $departed; background-color: $departed;
} }
&[data-status='stopped'] > &_train { &[data-status='stopped'] .user_train {
background-color: $stopped; background-color: $stopped;
} }
&[data-status='online'] > &_train { &[data-status='online'] .user_train {
background-color: $online; background-color: $online;
} }
&[data-status='terminated'] > &_train { &[data-status='terminated'] .user_train {
background-color: $terminated; background-color: $terminated;
} }
&[data-status='disconnected'] > &_train { &[data-status='disconnected'] .user_train {
background-color: $disconnected; background-color: $disconnected;
} }
@@ -139,6 +126,16 @@ ul {
pointer-events: none; pointer-events: none;
} }
} }
.user_train {
color: black;
background-color: $no-timetable;
transition: background-color 200ms;
-ms-transition: background-color 200ms;
-webkit-transition: background-color 200ms;
}
.users-anim { .users-anim {
&-move, &-move,
&-enter-active, &-enter-active,
+76 -43
View File
@@ -14,6 +14,10 @@
</span> </span>
<span class="header_links" v-if="station"> <span class="header_links" v-if="station">
<a :href="pragotronHref" target="_blank" :title="$t('scenery.pragotron-link')">
<img src="/images/icon-pragotron.svg" alt="icon-pragotron" />
</a>
<a :href="tabliceZbiorczeHref" target="_blank" :title="$t('scenery.tablice-link')"> <a :href="tabliceZbiorczeHref" target="_blank" :title="$t('scenery.tablice-link')">
<img src="/images/icon-tablice.ico" alt="icon-tablice" /> <img src="/images/icon-tablice.ico" alt="icon-tablice" />
</a> </a>
@@ -21,18 +25,15 @@
</h3> </h3>
<div class="timetable-checkpoints" v-if="station?.generalInfo?.checkpoints"> <div class="timetable-checkpoints" v-if="station?.generalInfo?.checkpoints">
<span v-for="(cp, i) in station.generalInfo.checkpoints" :key="i"> <template v-for="(ch, i) in station.generalInfo.checkpoints" :key="i">
{{ (i > 0 && '&bull;') || '' }} <template v-if="i > 0">&bull;</template>
<router-link
<button class="checkpoint-item"
:key="cp" :class="{ current: chosenCheckpoint === ch }"
class="checkpoint_item" :to="`/scenery?station=${station.name}&checkpoint=${ch}`"
:class="{ current: chosenCheckpoint === cp }" >{{ ch }}</router-link
@click="setCheckpoint(cp)"
> >
{{ cp }} </template>
</button>
</span>
</div> </div>
</div> </div>
@@ -62,18 +63,17 @@
{{ $t('scenery.no-timetables') }} {{ $t('scenery.no-timetables') }}
</div> </div>
<div <router-link
class="timetable-item" class="timetable-item a-block"
v-else v-else
v-for="(row, i) in sceneryTimetables" v-for="(row, i) in sceneryTimetables"
:key="row.train.id + i" :key="row.train.id + i"
tabindex="0" tabindex="0"
@click.prevent.stop="selectModalTrain(row.train, $event.currentTarget)" :to="row.train.driverRouteLocation"
@keydown.enter.prevent="selectModalTrain(row.train, $event.currentTarget)"
> >
<span class="timetable-general"> <span class="timetable-general">
<span class="general-info"> <span class="general-info">
<span> <div class="info-train">
<b <b
data-tooltip-type="BaseTooltip" data-tooltip-type="BaseTooltip"
:data-tooltip-content="getCategoryExplanation(row.train.timetableData!.category)" :data-tooltip-content="getCategoryExplanation(row.train.timetableData!.category)"
@@ -81,15 +81,18 @@
> >
{{ row.train.timetableData!.category }} {{ row.train.timetableData!.category }}
</b> </b>
<b>&nbsp;{{ row.train.trainNo }}</b> <span>&nbsp;</span>
<span v-if="row.checkpointStop.comments" :title="row.checkpointStop.comments"> <b>{{ row.train.trainNo }}</b>
<span>&nbsp;&bull;&nbsp;</span>
<span>{{ row.train.driverName }}</span>
<span
v-if="row.checkpointStop.comments"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="row.checkpointStop.comments"
>
<img src="/images/icon-warning.svg" /> <img src="/images/icon-warning.svg" />
</span> </span>
</span> </div>
&nbsp;&bull;&nbsp;
<span>
{{ row.train.driverName }}
</span>
<div class="info-route"> <div class="info-route">
<strong>{{ row.train.timetableData!.route.replace('|', ' - ') }}</strong> <strong>{{ row.train.timetableData!.route.replace('|', ' - ') }}</strong>
@@ -165,7 +168,7 @@
</span> </span>
</span> </span>
</span> </span>
</div> </router-link>
</transition-group> </transition-group>
</div> </div>
</section> </section>
@@ -178,21 +181,20 @@ import { useRoute } from 'vue-router';
import Loading from '../Global/Loading.vue'; import Loading from '../Global/Loading.vue';
import dateMixin from '../../mixins/dateMixin'; import dateMixin from '../../mixins/dateMixin';
import routerMixin from '../../mixins/routerMixin'; import routerMixin from '../../mixins/routerMixin';
import { useMainStore } from '../../store/mainStore';
import modalTrainMixin from '../../mixins/modalTrainMixin';
import ScheduledTrainStatus from './ScheduledTrainStatus.vue';
import { useApiStore } from '../../store/apiStore';
import { ActiveScenery, Station } from '../../typings/common';
import { SceneryTimetableRow } from './typings';
import { getTrainStopStatus, stopStatusPriority } from './utils';
import trainCategoryMixin from '../../mixins/trainCategoryMixin'; import trainCategoryMixin from '../../mixins/trainCategoryMixin';
import { useMainStore } from '../../store/mainStore';
import { useApiStore } from '../../store/apiStore';
import ScheduledTrainStatus from './ScheduledTrainStatus.vue';
import { SceneryTimetableRow } from './typings';
import { ActiveScenery, Station } from '../../typings/common';
import { getTrainStopStatus, stopStatusPriority } from './utils';
export default defineComponent({ export default defineComponent({
name: 'SceneryTimetable', name: 'SceneryTimetable',
components: { Loading, ScheduledTrainStatus }, components: { Loading, ScheduledTrainStatus },
mixins: [dateMixin, routerMixin, modalTrainMixin, trainCategoryMixin], mixins: [dateMixin, routerMixin, trainCategoryMixin],
props: { props: {
station: { station: {
@@ -211,6 +213,12 @@ export default defineComponent({
this.loadSelectedOption(); this.loadSelectedOption();
}, },
watch: {
currentURL() {
this.loadSelectedOption();
}
},
setup(props) { setup(props) {
const route = useRoute(); const route = useRoute();
const currentURL = computed(() => `${location.origin}${route.fullPath}`); const currentURL = computed(() => `${location.origin}${route.fullPath}`);
@@ -241,6 +249,13 @@ export default defineComponent({
return url; return url;
}, },
pragotronHref() {
let url = `https://pragotron-td2.web.app/board?name=${this.station!.name}&region=${this.mainStore.region.id}`;
if (this.chosenCheckpoint) url += `&checkpoint=${this.chosenCheckpoint}`;
return url;
},
sceneryTimetables(): SceneryTimetableRow[] { sceneryTimetables(): SceneryTimetableRow[] {
if (!this.onlineScenery) return []; if (!this.onlineScenery) return [];
@@ -292,7 +307,19 @@ export default defineComponent({
loadSelectedOption() { loadSelectedOption() {
if (!this.station) return; if (!this.station) return;
this.chosenCheckpoint = this.station.generalInfo?.checkpoints[0] ?? this.station.name; if (!this.station.generalInfo) {
this.chosenCheckpoint = this.station.name;
return;
}
const queryCheckpoint = this.$route.query['checkpoint']?.toString();
this.chosenCheckpoint =
this.station.generalInfo.checkpoints.find(
(ch) => ch.toLocaleLowerCase() === queryCheckpoint?.toLocaleLowerCase()
) ??
this.station.generalInfo.checkpoints[0] ??
this.station.name;
}, },
setCheckpoint(cp: string) { setCheckpoint(cp: string) {
@@ -362,7 +389,6 @@ export default defineComponent({
background: #353535; background: #353535;
cursor: pointer;
z-index: 10; z-index: 10;
&.empty { &.empty {
@@ -392,30 +418,35 @@ export default defineComponent({
} }
} }
.timetable-list {
position: relative;
}
.timetable-checkpoints { .timetable-checkpoints {
display: flex; display: flex;
justify-content: center; justify-content: center;
gap: 0.5em;
flex-wrap: wrap; flex-wrap: wrap;
font-size: 1.1em; font-size: 1.1em;
margin-top: 0.5em; margin-top: 0.5em;
}
button.checkpoint_item { .checkpoint-item {
color: #aaa; color: #aaa;
display: inline; display: inline;
&:hover {
color: white;
} }
.checkpoint_item.current { &.current {
font-weight: bold; font-weight: bold;
color: $accentCol; color: $accentCol;
} }
} }
.timetable-list {
position: relative;
}
.general-info { .general-info {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
@@ -429,7 +460,9 @@ export default defineComponent({
} }
img { img {
width: 1.1em; height: 0.9em;
vertical-align: middle;
margin: 0 0.25em;
} }
} }
@@ -6,30 +6,11 @@
<p>[F] {{ $t('options.filters') }}</p> <p>[F] {{ $t('options.filters') }}</p>
<span class="active-indicator" v-if="changedFilters.length != 0"></span> <span class="active-indicator" v-if="changedFilters.length != 0"></span>
</button> </button>
<label for="scenery-search">
<input
id="scenery-search"
list="sceneries"
:placeholder="$t('sceneries.scenery-search')"
@focus="preventKeyDown = true"
@blur="preventKeyDown = false"
v-model="chosenSearchScenery"
/>
<datalist id="sceneries">
<option
v-for="scenery in sortedStationList"
:key="scenery.name"
:value="scenery.name"
></option>
</datalist>
</label>
</div> </div>
<transition name="card-anim"> <transition name="card-anim">
<div class="card" v-if="isVisible" tabindex="0" ref="cardRef" @keydown.r="resetFilters"> <div class="card" v-if="isVisible" ref="cardRef" @keydown.r="resetFilters">
<div class="card_content" @scroll="onScroll" ref="cardContentRef"> <div class="card_content" tabindex="0" @scroll="onScroll" ref="cardContentRef">
<div class="card_title flex">{{ $t('filters.title') }}</div> <div class="card_title flex">{{ $t('filters.title') }}</div>
<p class="card_info" v-html="$t('filters.desc')"></p> <p class="card_info" v-html="$t('filters.desc')"></p>
@@ -40,6 +21,31 @@
<template v-else>{{ $t('filters.no-changed-filters') }}</template> <template v-else>{{ $t('filters.no-changed-filters') }}</template>
</div> </div>
<section class="card_sceneries-search">
<h3 class="section-header">{{ $t('filters.sceneries-search') }}</h3>
<datalist id="sceneries">
<option
v-for="scenery in sortedStationList"
:key="scenery.name"
:value="scenery.name"
></option>
</datalist>
<form action="javascript:void(0);" @submit="handleSceneriesInput">
<input
v-model="chosenSearchScenery"
id="scenery-search"
list="sceneries"
:placeholder="$t('filters.sceneries-placeholder')"
@focus="preventKeyDown = true"
@blur="preventKeyDown = false"
/>
<button class="btn--action">{{ $t('filters.search-button-title') }}</button>
</form>
</section>
<section class="card_options"> <section class="card_options">
<div <div
class="option-section" class="option-section"
@@ -57,16 +63,15 @@
<div class="section-filters"> <div class="section-filters">
<label <label
v-for="filterKey in sectionFilters" v-for="filterKey in sectionFilters"
@click="() => (filters[filterKey] = !filters[filterKey])"
@dblclick="setSingleSectionFilter(sectionKey, filterKey)" @dblclick="setSingleSectionFilter(sectionKey, filterKey)"
:for="filterKey"
> >
<input <input
type="checkbox"
:checked="filters[filterKey]" :checked="filters[filterKey]"
v-model="filters[filterKey]" v-model="filters[filterKey]"
type="checkbox"
:class="sectionKey" :class="sectionKey"
:name="filterKey" :name="filterKey"
:id="filterKey"
/> />
<span> <span>
{{ $t(`filters.${filterKey}`) }} {{ $t(`filters.${filterKey}`) }}
@@ -111,7 +116,7 @@
@blur="preventKeyDown = false" @blur="preventKeyDown = false"
/> />
<button class="btn--action">{{ $t('filters.authors-button-title') }}</button> <button class="btn--action">{{ $t('filters.search-button-title') }}</button>
</form> </form>
</section> </section>
@@ -269,20 +274,11 @@ export default defineComponent({
}, },
watch: { watch: {
chosenSearchScenery(value: string) {
const chosenStation = this.store.stationList.find(({ name }) => name == value);
if (chosenStation) {
this.$router.push(`/scenery?station=${chosenStation.name.replace(/ /g, '_')}`);
this.chosenSearchScenery = '';
}
},
isVisible(value: boolean) { isVisible(value: boolean) {
this.$nextTick(() => { this.$nextTick(() => {
if (value) { if (value) {
(this.$refs['cardRef'] as HTMLDivElement).focus();
(this.$refs['cardContentRef'] as HTMLDivElement).scrollTop = this.scrollTop; (this.$refs['cardContentRef'] as HTMLDivElement).scrollTop = this.scrollTop;
(this.$refs['cardContentRef'] as HTMLDivElement).focus();
} }
}); });
} }
@@ -300,7 +296,18 @@ export default defineComponent({
handleAuthorsInput() { handleAuthorsInput() {
this.filters['authors'] = this.authors; this.filters['authors'] = this.authors;
// if (this.saveOptions) StorageManager.setStringValue('authors', target.value); },
handleSceneriesInput() {
const chosenStation = this.store.stationList.find(
({ name }) => name == this.chosenSearchScenery
);
if (chosenStation) {
this.$router.push(`/scenery?station=${chosenStation.name.replace(/ /g, '_')}`);
this.chosenSearchScenery = '';
this.isVisible = false;
}
}, },
subHour() { subHour() {
@@ -329,6 +336,8 @@ export default defineComponent({
}, },
resetFilters() { resetFilters() {
if (this.preventKeyDown) return;
// Reset local model values // Reset local model values
this.minimumHours = 0; this.minimumHours = 0;
this.authors = ''; this.authors = '';
@@ -353,7 +362,8 @@ export default defineComponent({
setSingleSectionFilter(sectionKey: StationFilterSection, chosenKey: string) { setSingleSectionFilter(sectionKey: StationFilterSection, chosenKey: string) {
filtersSections[sectionKey].forEach((filterKey) => { filtersSections[sectionKey].forEach((filterKey) => {
if (filterKey != chosenKey) this.filters[filterKey] = initFilters[filterKey]; if (typeof this.filters[filterKey] === 'boolean')
this.filters[filterKey] = filterKey != chosenKey;
}); });
}, },
@@ -382,6 +392,7 @@ h3.section-header {
.card { .card {
display: grid; display: grid;
grid-template-rows: 1fr auto; grid-template-rows: 1fr auto;
padding: 1px;
} }
.card_info { .card_info {
@@ -451,8 +462,12 @@ h3.section-header {
} }
} }
.card_authors-search { .card_authors-search,
.card_sceneries-search {
margin: 1em 0; margin: 1em 0;
display: flex;
flex-direction: column;
align-items: center;
form { form {
display: flex; display: flex;
@@ -642,10 +657,6 @@ h3.section-header {
} }
@include smallScreen { @include smallScreen {
.card_controls > button.card-button > p {
display: none;
}
.slider { .slider {
flex-wrap: wrap; flex-wrap: wrap;
justify-content: center; justify-content: center;
+93 -68
View File
@@ -1,56 +1,70 @@
<template> <template>
<div class="station-stats"> <div
<div class="separator" /> class="dropdown"
@keydown.esc="showDropdown = false"
v-click-outside="() => (showDropdown = false)"
>
<div class="bg" v-if="showDropdown" @click="showDropdown = false"></div>
<div class="stats-row"> <button class="filter-button btn--filled btn--image" @click="toggleDropdown" ref="button">
<div> <img src="/images/icon-stats.svg" alt="Open filters icon" />
<span <!-- {{ $t('train-stats.stats-button') }} -->
>{{ $t('station-stats.u-factor') }} <span>STATYSTYKI</span>
<a </button>
href="https://td2.info.pl/dyskusje/wspolczynnik-ugla-czy-to-ma-sens/msg81011/#msg81011"
target="_blank"
:data-tooltip="$t('station-stats.u-factor-tooltip')"
>(?)</a
>:
</span>
<b class="u-factor" :style="calculateFactorStyle()"> <transition name="dropdown-anim">
{{ uFactor.toFixed(2) }} <div class="dropdown_wrapper" v-if="showDropdown">
</b> <div>
<h1 class="text--primary">
<img src="/images/icon-stats.svg" alt="Open filters icon" />
{{ $t('train-stats.title') }}
</h1>
<hr style="margin: 0.5em 0" />
<ul class="stats-list">
<li>
<span>
{{ $t('station-stats.u-factor') }}
<a
href="https://td2.info.pl/dyskusje/wspolczynnik-ugla-czy-to-ma-sens/msg81011/#msg81011"
target="_blank"
:data-tooltip="$t('station-stats.u-factor-tooltip')"
>(?)</a
>:
</span>
<b class="u-factor" :style="calculateFactorStyle()">
{{ uFactor.toFixed(2) }}
</b>
</li>
<li>
{{ $t('station-stats.avg-timetable-count') }}
<b>{{ avgTimetableCount.toFixed(2) }}</b>
</li>
<li>
{{ $t('station-stats.single-track-count') }}
<b>{{ trackCount.oneWay }}</b> (<b>{{ trackCount.oneWayElectric }} </b>)
</li>
<li>
{{ $t('station-stats.double-track-count') }}
<b>{{ trackCount.twoWay }}</b>
(<b>{{ trackCount.twoWayElectric }} </b>)
</li>
<li>
{{ $t('station-stats.cross-sceneries') }}
<b>{{ trackCount.crossTrack }}</b> (<b>{{ trackCount.crossTrackElectric }} </b>)
</li>
<li>
{{ $t('station-stats.open-spawns') }} <b>{{ spawnCount.passenger }}</b> - PAS /
<b>{{ spawnCount.freight }}</b> - TOW / <b>{{ spawnCount.loco }}</b> - LUZ /
<b>{{ spawnCount.all }}</b> - ALL
</li>
</ul>
</div>
<div tabindex="0" @focus="() => (showDropdown = false)"></div>
</div> </div>
</transition>
<div>
&bull;
{{ $t('station-stats.avg-timetable-count') }}
<b>{{ avgTimetableCount.toFixed(2) }}</b>
</div>
<div>
&bull;
{{ $t('station-stats.single-track-count') }}
<b>{{ trackCount.oneWay }}</b> (<b>{{ trackCount.oneWayElectric }} </b>)
</div>
<div>
&bull;
{{ $t('station-stats.double-track-count') }}
<b>{{ trackCount.twoWay }}</b>
(<b>{{ trackCount.twoWayElectric }} </b>)
</div>
<div>
&bull; {{ $t('station-stats.cross-sceneries') }} <b>{{ trackCount.crossTrack }}</b> (<b
>{{ trackCount.crossTrackElectric }} </b
>)
</div>
<div>
&bull;
{{ $t('station-stats.open-spawns') }} <b>{{ spawnCount.passenger }}</b> - PAS /
<b>{{ spawnCount.freight }}</b> - TOW / <b>{{ spawnCount.loco }}</b> - LUZ /
<b>{{ spawnCount.all }}</b> - ALL
</div>
</div>
</div> </div>
</template> </template>
@@ -61,11 +75,16 @@ import { useMainStore } from '../../store/mainStore';
export default defineComponent({ export default defineComponent({
data() { data() {
return { return {
mainStore: useMainStore() mainStore: useMainStore(),
showDropdown: false
}; };
}, },
methods: { methods: {
toggleDropdown() {
this.showDropdown = !this.showDropdown;
},
calculateFactorStyle() { calculateFactorStyle() {
if (this.uFactor == 0) return ''; if (this.uFactor == 0) return '';
@@ -171,25 +190,15 @@ export default defineComponent({
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
.separator { @import '../../styles/dropdown.scss';
width: 100%; @import '../../styles/badge.scss';
height: 2px;
h1 img {
vertical-align: text-bottom;
}
h3 {
margin: 0.5em 0; margin: 0.5em 0;
background-color: #aaa;
}
.station-stats {
text-align: center;
color: #ddd;
}
.stats-row {
display: flex;
justify-content: center;
flex-wrap: wrap;
text-wrap: pretty;
gap: 0.25em;
margin-top: 0.25em;
} }
.u-factor { .u-factor {
@@ -209,4 +218,20 @@ export default defineComponent({
color: rgb(22, 245, 22); color: rgb(22, 245, 22);
} }
} }
ul.stats-list {
list-style: disc;
padding-left: 1em;
margin-top: 1em;
& > li {
margin: 0.25em 0;
}
}
@include smallScreen {
.filter-button span {
display: none;
}
}
</style> </style>
+19 -24
View File
@@ -52,15 +52,14 @@
</thead> </thead>
<tbody> <tbody>
<tr <router-link
v-for="station in filteredStationList" v-for="station in filteredStationList"
:class="{ 'last-selected': lastSelectedStationName == station.name }" class="a-row"
role="row"
:key="station.name" :key="station.name"
@click.left="setScenery(station.name)" @click.right.prevent="openForumSite($event, station.generalInfo?.url)"
@click.right="openForumSite($event, station.generalInfo?.url)" @keydown.space.prevent="openForumSite($event, station.generalInfo?.url)"
@keydown.enter="setScenery(station.name)" :to="getSceneryRoute(station)"
@keydown.space="openForumSite($event, station.generalInfo?.url)"
tabindex="0"
> >
<td class="station-name" :class="station.generalInfo?.availability"> <td class="station-name" :class="station.generalInfo?.availability">
<b v-if="station.generalInfo?.project" style="color: salmon">{{ <b v-if="station.generalInfo?.project" style="color: salmon">{{
@@ -121,7 +120,7 @@
<span v-if="station.onlineInfo?.dispatcherName"> <span v-if="station.onlineInfo?.dispatcherName">
<b <b
v-if="apiStore.donatorsData.includes(station.onlineInfo.dispatcherName)" v-if="apiStore.donatorsData.includes(station.onlineInfo.dispatcherName)"
@click.stop="openDonationCard" @click.prevent="openDonationCard"
data-tooltip-type="DonatorTooltip" data-tooltip-type="DonatorTooltip"
:data-tooltip-content="$t('donations.dispatcher-message')" :data-tooltip-content="$t('donations.dispatcher-message')"
> >
@@ -294,7 +293,7 @@
> >
{{ station.onlineInfo?.scheduledTrainCount.confirmed ?? '-' }} {{ station.onlineInfo?.scheduledTrainCount.confirmed ?? '-' }}
</td> </td>
</tr> </router-link>
</tbody> </tbody>
</table> </table>
@@ -319,7 +318,7 @@ import dateMixin from '../../mixins/dateMixin';
import styleMixin from '../../mixins/styleMixin'; import styleMixin from '../../mixins/styleMixin';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
import { useMainStore } from '../../store/mainStore'; import { useMainStore } from '../../store/mainStore';
import { Status } from '../../typings/common'; import { Station, Status } from '../../typings/common';
import { useTooltipStore } from '../../store/tooltipStore'; import { useTooltipStore } from '../../store/tooltipStore';
import { getChangedFilters } from '../../managers/stationFilterManager'; import { getChangedFilters } from '../../managers/stationFilterManager';
import { ActiveSorter, HeadIdsType, headIconsIds, headIds } from './typings'; import { ActiveSorter, HeadIdsType, headIconsIds, headIds } from './typings';
@@ -334,7 +333,6 @@ export default defineComponent({
data: () => ({ data: () => ({
headIconsIds, headIconsIds,
headIds, headIds,
lastSelectedStationName: '',
getChangedFilters getChangedFilters
}), }),
@@ -364,21 +362,16 @@ export default defineComponent({
}, },
methods: { methods: {
setScenery(name: string) { getSceneryRoute(station: Station) {
const station = this.filteredStationList.find((station) => station.name === name); // TODO: Hide tooltips when navigating away
if (!station) return; return {
this.lastSelectedStationName = station.name;
this.tooltipStore.hide();
this.$router.push({
name: 'SceneryView', name: 'SceneryView',
query: { query: {
station: station.name.replaceAll(' ', '_'), station: station.name,
region: this.$route.query.region || undefined region: this.$route.query.region || undefined
} }
}); };
}, },
openDonationCard(e: Event) { openDonationCard(e: Event) {
@@ -414,9 +407,9 @@ export default defineComponent({
$rowCol: #424242; $rowCol: #424242;
.station_table { .station_table {
height: 80vh; height: calc(100vh - 11em);
max-height: 2000px; max-height: 2000px;
min-height: 700px; min-height: 500px;
overflow: auto; overflow: auto;
font-weight: 500; font-weight: 500;
} }
@@ -503,8 +496,10 @@ table {
} }
} }
tr { tr,
.a-row {
background-color: $rowCol; background-color: $rowCol;
vertical-align: middle;
&:nth-child(even) { &:nth-child(even) {
background-color: lighten($rowCol, 5); background-color: lighten($rowCol, 5);
+1 -1
View File
@@ -29,7 +29,7 @@ export default defineComponent({
border-radius: 0.25em; border-radius: 0.25em;
width: 100%; width: 100%;
background-color: #333; background-color: #1f1f1f;
box-shadow: 0 0 5px 2px #aaa; box-shadow: 0 0 5px 2px #aaa;
} }
+10 -8
View File
@@ -13,6 +13,8 @@ import BaseTooltip from './BaseTooltip.vue';
import SpawnsTooltip from './SpawnsTooltip.vue'; import SpawnsTooltip from './SpawnsTooltip.vue';
import UsersTooltip from './UsersTooltip.vue'; import UsersTooltip from './UsersTooltip.vue';
const BOX_PADDING_PX = 20;
export default defineComponent({ export default defineComponent({
components: { DonatorTooltip, VehiclePreviewTooltip, BaseTooltip, SpawnsTooltip, UsersTooltip }, components: { DonatorTooltip, VehiclePreviewTooltip, BaseTooltip, SpawnsTooltip, UsersTooltip },
@@ -33,14 +35,14 @@ export default defineComponent({
const boxWidth = previewEl.getBoundingClientRect().width; const boxWidth = previewEl.getBoundingClientRect().width;
let translateX = '0', let translateX = '0',
translateY = '30px'; translateY = `calc(-100% - ${BOX_PADDING_PX}px)`;
if (val[0] <= boxWidth / 2) { if (val[0] <= boxWidth / 2 + BOX_PADDING_PX) {
previewEl.style.left = '0'; previewEl.style.left = '0';
translateX = '0px'; translateX = BOX_PADDING_PX + 'px';
} else if (val[0] >= clientWidth - boxWidth / 2) { } else if (val[0] >= clientWidth - boxWidth / 2 - BOX_PADDING_PX) {
previewEl.style.left = '100%'; previewEl.style.left = '100%';
translateX = '-100%'; translateX = `calc(-100% - ${BOX_PADDING_PX}px)`;
} else { } else {
previewEl.style.left = `${val[0]}px`; previewEl.style.left = `${val[0]}px`;
translateX = '-50%'; translateX = '-50%';
@@ -49,10 +51,10 @@ export default defineComponent({
previewEl.style.top = `${val[1]}px`; previewEl.style.top = `${val[1]}px`;
const isOutside = const isOutside =
val[1] + previewEl.getBoundingClientRect().height + 30 >= val[1] - previewEl.getBoundingClientRect().height <=
window.innerHeight + window.scrollY; window.scrollY + BOX_PADDING_PX * 2;
if (isOutside) translateY = 'calc(-100% - 30px)'; if (isOutside) translateY = BOX_PADDING_PX + 'px';
previewEl.style.transform = `translate(${translateX}, ${translateY})`; previewEl.style.transform = `translate(${translateX}, ${translateY})`;
}); });
} }
@@ -89,7 +89,7 @@ export default defineComponent({
.tooltip-content { .tooltip-content {
width: 300px; width: 300px;
min-height: 200px; min-height: 200px;
background-color: #333; background-color: #1f1f1f;
box-shadow: 0 0 10px 2px #aaa; box-shadow: 0 0 10px 2px #aaa;
padding: 0.5em; padding: 0.5em;
+29 -15
View File
@@ -1,9 +1,13 @@
<template> <template>
<span <span
class="stop-label" class="stop-label"
:data-minor="stop.isSBL || (stop.nameRaw.endsWith(', po.') && !stop.duration)" :data-minor="stop.isSBL || (stop.nameRaw.endsWith(', po') && !stop.duration)"
> >
<span class="name" v-html="stop.nameHtml"></span> <router-link v-if="/(, podg$|<strong>)/.test(stop.nameHtml)" :to="sceneryHref">
<span class="stop-name" v-html="stop.nameHtml"></span>
</router-link>
<span v-else class="stop-name" v-html="stop.nameHtml"></span>
<span <span
v-if="stop.position != 'begin'" v-if="stop.position != 'begin'"
@@ -76,6 +80,12 @@ export default defineComponent({
type: Object as PropType<TrainScheduleStop>, type: Object as PropType<TrainScheduleStop>,
required: true required: true
} }
},
computed: {
sceneryHref() {
return `/scenery?station=${this.stop.sceneryName}&checkpoint=${this.stop.nameRaw}`;
}
} }
}); });
</script> </script>
@@ -97,19 +107,7 @@ s {
flex-wrap: wrap; flex-wrap: wrap;
align-items: center; align-items: center;
&[data-minor='true'] { .stop-name {
.date {
display: none;
}
.name {
background: none;
color: #aaa;
padding: 0;
}
}
.name {
background: $stopNameClr; background: $stopNameClr;
border-radius: 0.5em 0 0 0.5em; border-radius: 0.5em 0 0 0.5em;
padding: 0.3em 0.5em; padding: 0.3em 0.5em;
@@ -131,6 +129,18 @@ s {
} }
} }
&[data-minor='true'] {
.date {
display: none;
}
.stop-name {
background: none;
color: #aaa;
padding: 0;
}
}
.stop { .stop {
&[data-stop-types='ph'], &[data-stop-types='ph'],
&[data-stop-types='ph-pm'], &[data-stop-types='ph-pm'],
@@ -146,6 +156,10 @@ s {
} }
} }
.stop-label > a {
z-index: 0;
}
.stop .arrival { .stop .arrival {
&[data-status='confirmed'][data-status-delayed='true'] { &[data-status='confirmed'][data-status-delayed='true'] {
span { span {
+68 -72
View File
@@ -2,30 +2,28 @@
<div class="train-info" :data-extended="extended"> <div class="train-info" :data-extended="extended">
<section class="train-general"> <section class="train-general">
<div class="general-top-bar"> <div class="general-top-bar">
<div> <div class="top-bar-header">
<b class="warning-timeout" v-if="train.isTimeout" :title="$t('trains.timeout')">?</b> <b class="warning-timeout" v-if="train.isTimeout" :title="$t('trains.timeout')">?</b>
<span class="timetable-id" v-if="train.timetableData"> <span class="timetable-id" v-if="train.timetableData">
#{{ train.timetableData.timetableId }} #{{ train.timetableData.timetableId }}
</span> </span>
<span <span
class="timetable-warnings" class="train-badge twr"
v-if="train.timetableData?.TWR || train.timetableData?.SKR" v-if="train.timetableData?.TWR"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('general.TWR') + `:\n${train.timetableData.warningNotes}`"
> >
<span TWR
class="train-badge twr" </span>
v-if="train.timetableData?.TWR"
:title="$t('general.TWR')" <span
> class="train-badge skr"
TWR v-if="train.timetableData?.SKR"
</span> data-tooltip-type="BaseTooltip"
<span :data-tooltip-content="$t('general.SKR')"
class="train-badge skr" >
v-if="train.timetableData?.SKR" SKR
:title="$t('general.SKR')"
>
SKR
</span>
</span> </span>
<b <b
@@ -38,14 +36,15 @@
</b> </b>
<b class="train-number">{{ train.trainNo }}</b> <b class="train-number">{{ train.trainNo }}</b>
<span>&bull;</span> <span>&bull;</span>
<b
class="level-badge driver"
:style="calculateExpStyle(train.driverLevel, train.isSupporter)"
>
{{ train.driverLevel < 2 ? 'L' : `${train.driverLevel}` }}
</b>
<div class="train-driver"> <div class="train-driver">
<b
class="level-badge driver"
:style="calculateExpStyle(train.driverLevel, train.isSupporter)"
>
{{ train.driverLevel < 2 ? 'L' : `${train.driverLevel}` }}
</b>
<b <b
v-if="apiStore.donatorsData.includes(train.driverName)" v-if="apiStore.donatorsData.includes(train.driverName)"
data-tooltip-type="DonatorTooltip" data-tooltip-type="DonatorTooltip"
@@ -58,19 +57,6 @@
<span v-else>{{ train.driverName }}</span> <span v-else>{{ train.driverName }}</span>
</div> </div>
</div> </div>
<div v-if="extended">
<button class="btn-timetable btn--image btn--action" @click="navigateToJournal">
<img src="/images/icon-train.svg" alt="train icon" />
<span>
{{ $t('trains.journal-button') }}
</span>
</button>
<button class="btn-exit btn--image btn--action" @click="closeModal">
<img src="/images/icon-exit.svg" alt="modal exit icon" />
</button>
</div>
</div> </div>
<div class="general-timetable" v-if="train.timetableData"> <div class="general-timetable" v-if="train.timetableData">
@@ -91,7 +77,7 @@
<div class="general-stops" v-if="train.timetableData"> <div class="general-stops" v-if="train.timetableData">
<span v-if="train.timetableData.followingStops.length > 2"> <span v-if="train.timetableData.followingStops.length > 2">
{{ $t('trains.via-title') }} {{ $t('trains.via-title') }}
<span v-html="displayStopList(train.timetableData.followingStops)"></span> <span v-html="getTrainStopsHtml(train.timetableData.followingStops)"></span>
</span> </span>
</div> </div>
@@ -100,21 +86,22 @@
<ProgressBar :progressPercent="confirmedPercentage(train.timetableData.followingStops)" /> <ProgressBar :progressPercent="confirmedPercentage(train.timetableData.followingStops)" />
<span class="progress-distance"> <span class="progress-distance">
&nbsp; {{ currentDistance(train.timetableData.followingStops) }} km / <span>{{ currentDistance(train.timetableData.followingStops) }} km</span>
<span class="text--primary"> {{ train.timetableData.routeDistance }} km </span> <span>/</span>
| <span class="text--primary">{{ train.timetableData.routeDistance }} km </span>
<span>|</span>
<span v-html="currentDelay(train.timetableData.followingStops)"></span> <span v-html="currentDelay(train.timetableData.followingStops)"></span>
</span> </span>
</div> </div>
<div class="status-badges"> <div class="status-badges">
<div v-if="!train.currentStationHash" class="train-badge offline"> <div v-if="!train.currentStationHash" class="train-badge offline">
<img src="/images/icon-offline.svg" alt="" /> <img src="/images/icon-offline.svg" alt="offline train icon" />
{{ $t('trains.scenery-offline') }} {{ $t('trains.scenery-offline') }}
</div> </div>
<div v-if="!train.online" class="train-badge offline"> <div v-if="!train.online" class="train-badge offline">
<img src="/images/icon-offline.svg" alt="" /> <img src="/images/icon-offline.svg" alt="offline train icon" />
Offline {{ lastSeenMessage(train.lastSeen) }} Offline {{ lastSeenMessage(train.lastSeen) }}
</div> </div>
</div> </div>
@@ -152,6 +139,20 @@
<div class="text--grayed" style="margin-top: 0.25em"> <div class="text--grayed" style="margin-top: 0.25em">
{{ displayTrainPosition(train) }} {{ displayTrainPosition(train) }}
</div> </div>
<div
class="train-dangers"
v-if="extended && (train.timetableData?.TWR || train.timetableData?.SKR)"
>
<div v-if="train.timetableData.TWR">
<b style="color: var(--clr-twr)">TWR</b> - {{ $t('general.TWR') }}
<i>({{ train.timetableData?.warningNotes }})</i>
</div>
<div v-if="train.timetableData.SKR">
<b style="color: var(--clr-skr)">SKR</b> - {{ $t('general.SKR') }}
</div>
</div>
</section> </section>
<section class="train-stats" v-if="!extended"> <section class="train-stats" v-if="!extended">
@@ -182,12 +183,11 @@ import ProgressBar from '../Global/ProgressBar.vue';
import { useMainStore } from '../../store/mainStore'; import { useMainStore } from '../../store/mainStore';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
import StockList from '../Global/StockList.vue'; import StockList from '../Global/StockList.vue';
import modalTrainMixin from '../../mixins/modalTrainMixin';
import { Train } from '../../typings/common'; import { Train } from '../../typings/common';
import trainCategoryMixin from '../../mixins/trainCategoryMixin'; import trainCategoryMixin from '../../mixins/trainCategoryMixin';
export default defineComponent({ export default defineComponent({
mixins: [trainInfoMixin, styleMixin, modalTrainMixin, trainCategoryMixin], mixins: [trainInfoMixin, styleMixin, trainCategoryMixin],
components: { ProgressBar, StockList }, components: { ProgressBar, StockList },
props: { props: {
@@ -216,19 +216,14 @@ export default defineComponent({
return Math.min(vehicleSpeed, acc); return Math.min(vehicleSpeed, acc);
}, 300); }, 300);
} },
}, journalRouteLocation() {
return {
methods: {
navigateToJournal() {
this.$router.push({
path: '/journal/timetables', path: '/journal/timetables',
query: { query: {
'search-driver': this.train.driverName 'search-driver': this.train.driverName
} }
}); };
this.closeModal();
} }
} }
}); });
@@ -255,6 +250,10 @@ export default defineComponent({
line-height: 1.5em; line-height: 1.5em;
} }
.train-dangers {
margin-top: 0.5em;
}
.train-info { .train-info {
display: grid; display: grid;
grid-template-columns: 2fr 1fr; grid-template-columns: 2fr 1fr;
@@ -270,6 +269,12 @@ export default defineComponent({
gap: 0.5em; gap: 0.5em;
} }
.train-driver {
display: flex;
align-items: center;
gap: 0.25em;
}
.train-driver img { .train-driver img {
max-height: 20px; max-height: 20px;
vertical-align: text-bottom; vertical-align: text-bottom;
@@ -301,24 +306,15 @@ export default defineComponent({
.general-top-bar { .general-top-bar {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
flex-wrap: wrap;
gap: 0.5em; gap: 0.5em;
& > div {
display: flex;
align-items: center;
flex-wrap: wrap;
gap: 0.25em;
}
} }
.btn-timetable { .top-bar-header {
padding: 0.25em; display: flex;
} align-items: center;
flex-wrap: wrap;
.btn-exit { gap: 0.25em;
padding: 0.25em;
} }
.general-status { .general-status {
@@ -365,10 +361,14 @@ export default defineComponent({
display: flex; display: flex;
align-items: center; align-items: center;
flex-wrap: wrap; flex-wrap: wrap;
gap: 0.5em;
padding: 0.5em 0;
} }
.progress-distance { .progress-distance {
margin-right: 0.25em; display: flex;
flex-wrap: wrap;
gap: 0.25em;
} }
.timetable-warnings { .timetable-warnings {
@@ -381,9 +381,5 @@ export default defineComponent({
grid-template-columns: 1fr; grid-template-columns: 1fr;
gap: 1em 0; gap: 1em 0;
} }
.btn-timetable > span {
display: none;
}
} }
</style> </style>
-103
View File
@@ -1,103 +0,0 @@
<template>
<div class="train-modal" v-if="chosenTrain" @keydown.esc="closeModal">
<div class="modal-background" @click="closeModal"></div>
<div class="modal-content" ref="content" tabindex="0">
<TrainInfo :train="chosenTrain" :extended="true" ref="trainInfo" />
<TrainSchedule :train="chosenTrain" tabindex="0" />
</div>
</div>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import modalTrainMixin from '../../mixins/modalTrainMixin';
import TrainInfo from './TrainInfo.vue';
import TrainSchedule from './TrainSchedule.vue';
import { Train } from '../../typings/common';
export default defineComponent({
components: { TrainInfo, TrainSchedule },
mixins: [modalTrainMixin],
computed: {
chosenTrain() {
return this.store.trainList.find((train) => train.modalId == this.store.chosenModalTrainId);
}
},
watch: {
chosenTrain(train: Train | undefined) {
this.$nextTick(() => {
if (train) {
document.body.classList.add('no-scroll');
const contentEl = this.$refs['content'] as HTMLElement;
contentEl.focus();
} else {
(this.store.modalLastClickedTarget as any)?.focus();
setTimeout(() => {
document.body.classList.remove('no-scroll');
}, 90);
}
});
}
}
});
</script>
<style lang="scss" scoped>
@import '../../styles/responsive.scss';
.train-modal {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
color: white;
z-index: 200;
display: flex;
justify-content: center;
align-items: flex-start;
text-align: left;
}
.modal-background {
position: absolute;
top: 0;
left: 0;
width: 100vw;
height: 100vh;
cursor: pointer;
background-color: rgba(0, 0, 0, 0.55);
}
.modal-content {
position: relative;
overflow-y: scroll;
width: 95vw;
max-height: 95vh;
max-height: 95dvh;
margin-top: 1em;
background-color: #1a1a1a;
box-shadow: 0 0 15px 10px #0e0e0e;
}
@include midScreen {
.exit {
margin: 0.5em;
img {
width: 1.75rem;
}
}
}
</style>
+6 -2
View File
@@ -15,12 +15,14 @@
<div class="search_content"> <div class="search_content">
<div class="search-box"> <div class="search-box">
<input <input
v-model="searchedTrain"
class="search-input" class="search-input"
ref="initFocusedElement" ref="initFocusedElement"
id="train-search"
name="train-search"
@focus="preventKeyDown = true" @focus="preventKeyDown = true"
@blur="preventKeyDown = false" @blur="preventKeyDown = false"
:placeholder="$t(`options.search-train`)" :placeholder="$t(`options.search-train`)"
v-model="searchedTrain"
/> />
<button class="search-exit"> <button class="search-exit">
<img <img
@@ -33,11 +35,13 @@
<div class="search-box"> <div class="search-box">
<input <input
v-model="searchedDriver"
class="search-input" class="search-input"
id="driver-search"
name="driver-search"
@focus="preventKeyDown = true" @focus="preventKeyDown = true"
@blur="preventKeyDown = false" @blur="preventKeyDown = false"
:placeholder="$t(`options.search-driver`)" :placeholder="$t(`options.search-driver`)"
v-model="searchedDriver"
/> />
<button class="search-exit"> <button class="search-exit">
<img <img
+23 -14
View File
@@ -60,7 +60,7 @@
v-else v-else
src="/images/icon-we4a.png" src="/images/icon-we4a.png"
:title="$t('trains.we4a-tooltip')" :title="$t('trains.we4a-tooltip')"
width="12" width="10"
/> />
</span> </span>
</div> </div>
@@ -84,7 +84,7 @@
v-else v-else
src="/images/icon-we4a.png" src="/images/icon-we4a.png"
:title="$t('trains.we4a-tooltip')" :title="$t('trains.we4a-tooltip')"
width="12" width="10"
/> />
</span> </span>
</div> </div>
@@ -165,20 +165,18 @@ export default defineComponent({
computed: { computed: {
scheduleStops(): TrainScheduleStop[] { scheduleStops(): TrainScheduleStop[] {
let currentSceneryIndex = 0; if (!this.train.timetableData) return [];
const { timetablePath } = this.train.timetableData;
let currentPathIndex = 0;
return ( return (
this.train.timetableData?.followingStops.map((stop, i, arr) => { this.train.timetableData?.followingStops.map((stop, i, arr) => {
const isExternal = const isExternal =
i > 0 && i < arr.length - 1 &&
stop.arrivalLine != null && stop.departureLine === timetablePath[currentPathIndex].departureRouteExt;
(stop.arrivalLine != arr[i - 1].departureLine ||
(stop.arrivalLine == arr[i - 1].departureLine &&
!/-|_|(^it\d+)|(^sbl)/gi.test(stop.arrivalLine)));
if (isExternal) currentSceneryIndex++; const sceneryName = timetablePath[currentPathIndex].stationName;
const sceneryName = this.train.timetableData!.sceneryNames[currentSceneryIndex];
const sceneryInfo = this.apiStore.sceneryData.find((st) => st.name == sceneryName); const sceneryInfo = this.apiStore.sceneryData.find((st) => st.name == sceneryName);
const arrivalLineInfo = sceneryInfo?.routesInfo.find( const arrivalLineInfo = sceneryInfo?.routesInfo.find(
@@ -189,6 +187,8 @@ export default defineComponent({
(r) => r.routeName == stop.departureLine (r) => r.routeName == stop.departureLine
); );
if (isExternal) currentPathIndex++;
return { return {
nameHtml: stop.stopName, nameHtml: stop.stopName,
nameRaw: stop.stopNameRAW, nameRaw: stop.stopNameRAW,
@@ -249,7 +249,7 @@ export default defineComponent({
i < this.train.timetableData!.followingStops.length; i < this.train.timetableData!.followingStops.length;
i++ i++
) { ) {
if (/po\.|sbl/gi.test(this.train.timetableData!.followingStops[i].stopNameRAW)) if (/(, po$|sbl|, pe$)/gi.test(this.train.timetableData!.followingStops[i].stopNameRAW))
activeMinorStopList.push(i); activeMinorStopList.push(i);
else break; else break;
} }
@@ -286,7 +286,7 @@ $blinkAnim: 0.5s ease-in-out alternate infinite blink;
} }
.train-schedule { .train-schedule {
padding: 0 1em; padding: 1em;
} }
.schedule-wrapper { .schedule-wrapper {
@@ -523,8 +523,17 @@ $blinkAnim: 0.5s ease-in-out alternate infinite blink;
} }
.scenery-route { .scenery-route {
display: flex;
gap: 0.25em;
span:nth-child(2) {
display: flex;
gap: 0.25em;
align-items: center;
}
img { img {
vertical-align: middle; width: 1em;
} }
} }
+13 -38
View File
@@ -1,5 +1,9 @@
<template> <template>
<div class="dropdown" @keydown.esc="showOptions = false" @focusout="showOptions = false"> <div
class="dropdown"
@keydown.esc="showOptions = false"
v-click-outside="() => (showOptions = false)"
>
<div class="bg" v-if="showOptions" @click="showOptions = false"></div> <div class="bg" v-if="showOptions" @click="showOptions = false"></div>
<button class="filter-button btn--filled btn--image" @click="toggleShowOptions" ref="button"> <button class="filter-button btn--filled btn--image" @click="toggleShowOptions" ref="button">
@@ -19,21 +23,21 @@
<div v-if="apiStore.dataStatuses.connection == Status.Loaded && regionTrains.length > 0"> <div v-if="apiStore.dataStatuses.connection == Status.Loaded && regionTrains.length > 0">
<div class="top-list general"> <div class="top-list general">
<transition-group tag="ul" name="stats-anim"> <transition-group tag="ul" name="stats-anim">
<li class="badge" key="timetable-count"> <li class="badge stat-badge" key="timetable-count">
<span>{{ $t('train-stats.timetable-count') }}</span> <span>{{ $t('train-stats.timetable-count') }}</span>
<span> <span>
<b>{{ regionTrainsWithTT.length }}</b> <b>{{ regionTrainsWithTT.length }}</b>
</span> </span>
</li> </li>
<li class="badge" key="avg-speed"> <li class="badge stat-badge" key="avg-speed">
<span>{{ $t('train-stats.avg-speed') }}</span> <span>{{ $t('train-stats.avg-speed') }}</span>
<span> <span>
<b>{{ stats.avgSpeed.toFixed(1) }} km/h</b> <b>{{ stats.avgSpeed.toFixed(1) }} km/h</b>
</span> </span>
</li> </li>
<li class="badge" key="avg-distance"> <li class="badge stat-badge" key="avg-distance">
<span>{{ $t('train-stats.avg-timetable') }}</span> <span>{{ $t('train-stats.avg-timetable') }}</span>
<span> <span>
<b>{{ stats.avgDistance.toFixed(1) }} km</b> <b>{{ stats.avgDistance.toFixed(1) }} km</b>
@@ -46,7 +50,7 @@
<h3>{{ $t('train-stats.top-categories') }}</h3> <h3>{{ $t('train-stats.top-categories') }}</h3>
<transition-group tag="ul" name="stats-anim"> <transition-group tag="ul" name="stats-anim">
<li class="badge" v-for="top in stats.topCategories" :key="top.name"> <li class="badge stat-badge" v-for="top in stats.topCategories" :key="top.name">
<span>{{ top.name }}</span> <span>{{ top.name }}</span>
<span>{{ top.count }}</span> <span>{{ top.count }}</span>
</li> </li>
@@ -61,7 +65,7 @@
<h3>{{ $t('train-stats.top-vehicles') }}</h3> <h3>{{ $t('train-stats.top-vehicles') }}</h3>
<transition-group tag="ul" name="stats-anim"> <transition-group tag="ul" name="stats-anim">
<li class="badge" v-for="top in stats.topVehicles" :key="top.name"> <li class="badge stat-badge" v-for="top in stats.topVehicles" :key="top.name">
<span>{{ top.name }}</span> <span>{{ top.name }}</span>
<span>{{ top.count }}</span> <span>{{ top.count }}</span>
</li> </li>
@@ -76,7 +80,7 @@
<h3>{{ $t('train-stats.top-units') }}</h3> <h3>{{ $t('train-stats.top-units') }}</h3>
<transition-group tag="ul" name="stats-anim"> <transition-group tag="ul" name="stats-anim">
<li class="badge" v-for="top in stats.topUnits.slice(0, 7)" :key="top.name"> <li class="badge stat-badge" v-for="top in stats.topUnits.slice(0, 7)" :key="top.name">
<span>{{ top.name }}</span> <span>{{ top.name }}</span>
<span>{{ top.count }}</span> <span>{{ top.count }}</span>
</li> </li>
@@ -95,6 +99,8 @@
<div class="no-data" v-else> <div class="no-data" v-else>
{{ $t('train-stats.no-stats') }} {{ $t('train-stats.no-stats') }}
</div> </div>
<div tabindex="0" @focus="() => (showOptions = false)"></div>
</div> </div>
</transition> </transition>
</div> </div>
@@ -236,43 +242,12 @@ h3 {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
gap: 0.5em; gap: 0.5em;
// @include smallScreen {
// justify-content: center;
// }
}
.badge {
margin: 0;
& > span:first-child {
background-color: $accentCol;
color: black;
}
} }
.dropdown_wrapper { .dropdown_wrapper {
max-width: 600px; max-width: 600px;
} }
.stats-anim {
&-move,
&-enter-active,
&-leave-active {
transition: all 250ms ease;
}
&-enter-from,
&-leave-to {
opacity: 0;
transform: translateX(5px);
}
&-leave-active {
position: absolute;
}
}
@include smallScreen { @include smallScreen {
h1, h1,
.no-data { .no-data {
+6 -12
View File
@@ -17,11 +17,10 @@
class="train-row" class="train-row"
v-for="train in trains" v-for="train in trains"
:key="train.id" :key="train.id"
tabindex="0"
@click.stop="selectModalTrain(train, $event.currentTarget)"
@keydown.enter="selectModalTrain(train, $event.currentTarget)"
> >
<TrainInfo :train="train" :extended="false" /> <router-link class="a-block" :to="train.driverRouteLocation">
<TrainInfo :train="train" :extended="false" />
</router-link>
</li> </li>
</transition-group> </transition-group>
</div> </div>
@@ -30,7 +29,6 @@
<script lang="ts"> <script lang="ts">
import { defineComponent, inject, PropType, Ref } from 'vue'; import { defineComponent, inject, PropType, Ref } from 'vue';
import modalTrainMixin from '../../mixins/modalTrainMixin';
import { useMainStore } from '../../store/mainStore'; import { useMainStore } from '../../store/mainStore';
import Loading from '../Global/Loading.vue'; import Loading from '../Global/Loading.vue';
import TrainInfo from './TrainInfo.vue'; import TrainInfo from './TrainInfo.vue';
@@ -47,8 +45,6 @@ export default defineComponent({
} }
}, },
mixins: [modalTrainMixin],
setup() { setup() {
const store = useMainStore(); const store = useMainStore();
const apiStore = useApiStore(); const apiStore = useApiStore();
@@ -86,10 +82,10 @@ export default defineComponent({
@import '../../styles/animations.scss'; @import '../../styles/animations.scss';
.train-table { .train-table {
position: relative; height: calc(100vh - 11em);
min-height: 500px;
height: 90vh; position: relative;
min-height: 550px;
overflow-y: auto; overflow-y: auto;
overflow-x: hidden; overflow-x: hidden;
} }
@@ -107,7 +103,5 @@ li.train-row {
background-color: var(--clr-secondary); background-color: var(--clr-secondary);
margin-bottom: 1em; margin-bottom: 1em;
width: 100%; width: 100%;
cursor: pointer;
} }
</style> </style>
+23 -4
View File
@@ -83,6 +83,9 @@
"ZN": "inspection / diagnostic type", "ZN": "inspection / diagnostic type",
"ZU": "other maintenance type", "ZU": "other maintenance type",
"ZG": "emergency (deprecated)",
"AP": "voivodeship regio (deprecated)",
"E": "electric loco", "E": "electric loco",
"J": "EMU", "J": "EMU",
@@ -292,9 +295,11 @@
"minTwoWay": "MIN. OTHER DOUBLE TRACK ROUTES" "minTwoWay": "MIN. OTHER DOUBLE TRACK ROUTES"
}, },
"sceneries-search": "SCENERY SEARCH:",
"sceneries-placeholder": "Enter scenery name...",
"authors-search": "SEARCH BY AUTHOR NAME (other filters apply):", "authors-search": "SEARCH BY AUTHOR NAME (other filters apply):",
"authors-placeholder": "Enter the author nickname...", "authors-placeholder": "Enter the author nickname...",
"authors-button-title": "Search", "search-button-title": "SEARCH",
"minimum-hours-title": "SHOW ONLY SCENERIES UNTIL:", "minimum-hours-title": "SHOW ONLY SCENERIES UNTIL:",
"now": "NOW", "now": "NOW",
@@ -400,7 +405,15 @@
"scenery-offline": "Offline ride", "scenery-offline": "Offline ride",
"timeout": "An error occured while trying to refresh SWDR timetable data!", "timeout": "An error occured while trying to refresh SWDR timetable data!",
"journal-button": "DRIVER'S JOURNAL" "driver-journal-link": "DRIVER JOURNAL",
"driver-return-link": "GO BACK",
"driver-not-found-header": "Train not found! :/",
"driver-not-found-desc-1": "This train has already been terminated, changed its number or is offline.",
"driver-not-found-desc-2": "You can browse timetable history in the",
"driver-not-found-journal": "TIMETABLES JOURNAL",
"driver-not-found-others": "Player {driver} is online as:",
"driver-not-found-return": "GO BACK TO THE MAIN SITE"
}, },
"train-stats": { "train-stats": {
"stats-button": "STATISTICS", "stats-button": "STATISTICS",
@@ -429,6 +442,7 @@
"no-further-data": "No further data for current parameters", "no-further-data": "No further data for current parameters",
"loading-further-data": "Loading...", "loading-further-data": "Loading...",
"route-length": "Route length:", "route-length": "Route length:",
"station-count": "Stations:", "station-count": "Stations:",
"dispatcher-name": "Author", "dispatcher-name": "Author",
@@ -445,11 +459,16 @@
"minutes": "{value} min | {value} mins", "minutes": "{value} min | {value} mins",
"seconds": "{value} s", "seconds": "{value} s",
"stock-info": "DETAILS", "entry-details": "DETAILS",
"no-entry-details": "NO DETAILS AVAILABLE",
"stock-length": "Length", "stock-length": "Length",
"stock-mass": "Mass", "stock-mass": "Mass",
"stock-max-speed": "Max. speed", "stock-max-speed": "Max. speed",
"stock-dangers": "ADDITIONAL NOTES",
"stock-preview": "STOCK PREVIEW",
"load-data": "Load further data...", "load-data": "Load further data...",
"last-seen-at": "Last seen at", "last-seen-at": "Last seen at",
@@ -552,7 +571,7 @@
"forum-topic": "Official {name} forum topic", "forum-topic": "Official {name} forum topic",
"pragotron-link": "Timetable pallet board (beta)", "pragotron-link": "Timetable pallet board",
"tablice-link": "Timetable summary board (by Thundo)", "tablice-link": "Timetable summary board (by Thundo)",
"bottom-info": "Show full history in the Journal tab" "bottom-info": "Show full history in the Journal tab"
+23 -5
View File
@@ -80,6 +80,9 @@
"ZN": "inspekcyjny / diagnostyczny", "ZN": "inspekcyjny / diagnostyczny",
"ZU": "inny utrzymaniowy", "ZU": "inny utrzymaniowy",
"ZG": "ratunkowy (kat. wycofana)",
"AP": "wojewódzki osobowy (kat. wycofana)",
"E": "elektrowóz", "E": "elektrowóz",
"J": "EZT", "J": "EZT",
@@ -290,9 +293,11 @@
"minTwoWay": "SZLAKI DWUTOROWE NIEZELEKTR. (MINIMUM)" "minTwoWay": "SZLAKI DWUTOROWE NIEZELEKTR. (MINIMUM)"
}, },
"authors-search": "SZUKAJ AUTORA (uwzględnia inne filtry):", "sceneries-search": "WYSZUKAJ SCENERIĘ:",
"sceneries-placeholder": "Wpisz nazwę scenerii...",
"authors-search": "WYSZUKAJ AUTORA (uwzględnia inne filtry):",
"authors-placeholder": "Wpisz nick autora...", "authors-placeholder": "Wpisz nick autora...",
"authors-button-title": "Szukaj", "search-button-title": "SZUKAJ",
"minimum-hours-title": "POKAŻ TYLKO SCENERIE DOSTĘPNE MINIMUM DO:", "minimum-hours-title": "POKAŻ TYLKO SCENERIE DOSTĘPNE MINIMUM DO:",
"now": "TERAZ", "now": "TERAZ",
"hour": " godz.", "hour": " godz.",
@@ -386,7 +391,15 @@
"timeout": "Wystąpił problem z aktualizacją rozkładów jazdy z SWDR", "timeout": "Wystąpił problem z aktualizacją rozkładów jazdy z SWDR",
"journal-button": "DZIENNIK MASZYNISTY" "driver-journal-link": "DZIENNIK MASZYNISTY",
"driver-return-link": "POWRÓT",
"driver-not-found-header": "Nie znaleziono pociągu! :/",
"driver-not-found-desc-1": "Ten pociąg prawdopodobnie zakończył już swój bieg, zmienił numer lub jest offline.",
"driver-not-found-desc-2": "Historię rozkładów jazdy możesz przejrzeć w",
"driver-not-found-journal": "DZIENNIKU RJ",
"driver-not-found-others": "Gracz {driver} jest online jako:",
"driver-not-found-return": "WRÓĆ NA STRONĘ GŁÓWNĄ"
}, },
"train-stats": { "train-stats": {
"stats-button": "STATYSTYKI", "stats-button": "STATYSTYKI",
@@ -430,11 +443,16 @@
"timetable-abandoned": "PORZUCONY", "timetable-abandoned": "PORZUCONY",
"timetable-online-button": "RJ ONLINE", "timetable-online-button": "RJ ONLINE",
"stock-info": "SZCZEGÓŁY", "entry-details": "SZCZEGÓŁY",
"no-entry-details": "BRAK DOSTĘPNYCH SZCZEGÓŁÓW",
"stock-length": "Długość", "stock-length": "Długość",
"stock-mass": "Masa", "stock-mass": "Masa",
"stock-max-speed": "Prędkość maks.", "stock-max-speed": "Prędkość maks.",
"stock-dangers": "DODATKOWE UWAGI",
"stock-preview": "PODGLĄD SKŁADU",
"load-data": "Pobierz dalszą historię...", "load-data": "Pobierz dalszą historię...",
"last-seen-at": "Ostatnio widziany na: ", "last-seen-at": "Ostatnio widziany na: ",
@@ -536,7 +554,7 @@
"forum-topic": "Oficjalny wątek scenerii {name}", "forum-topic": "Oficjalny wątek scenerii {name}",
"pragotron-link": "Paletowa tablica informacyjna (beta)", "pragotron-link": "Paletowa tablica informacyjna",
"tablice-link": "Tablica informacyjna zbiorcza (autorstwa Thundo)", "tablice-link": "Tablica informacyjna zbiorcza (autorstwa Thundo)",
"bottom-info": "Pokaż pełną historię w zakładce Dziennika" "bottom-info": "Pokaż pełną historię w zakładce Dziennika"
+6 -2
View File
@@ -57,6 +57,10 @@ export default defineComponent({
: ''; : '';
}, },
dateStringToTimestamp(dateString?: string) {
return dateString ? new Date(dateString).getTime() : 0;
},
calculateDuration(timestampMs: number, showSeconds = false) { calculateDuration(timestampMs: number, showSeconds = false) {
const secondsTotal = Math.floor(timestampMs / 1000); const secondsTotal = Math.floor(timestampMs / 1000);
const minsTotal = Math.round(timestampMs / 60000); const minsTotal = Math.round(timestampMs / 60000);
@@ -70,8 +74,8 @@ export default defineComponent({
minsInHour minsInHour
)}` )}`
: showSeconds && secondsTotal <= 60 : showSeconds && secondsTotal <= 60
? this.$t('journal.seconds', { value: secondsTotal }, secondsTotal) ? this.$t('journal.seconds', { value: secondsTotal }, secondsTotal)
: this.$t('journal.minutes', { value: minsTotal }, minsTotal); : this.$t('journal.minutes', { value: minsTotal }, minsTotal);
} }
} }
}); });
-30
View File
@@ -1,30 +0,0 @@
import { defineComponent } from 'vue';
import { useMainStore } from '../store/mainStore';
import { useTooltipStore } from '../store/tooltipStore';
import { Train } from '../typings/common';
export default defineComponent({
data() {
return {
store: useMainStore(),
tooltipStore: useTooltipStore()
};
},
methods: {
selectModalTrain(train: Train, target?: EventTarget | null) {
this.store.chosenModalTrainId = train.modalId;
if (target) this.store.modalLastClickedTarget = target;
},
selectModalTrainById(modalId: string, target?: EventTarget | null) {
this.store.chosenModalTrainId = modalId;
if (target) this.store.modalLastClickedTarget = target;
},
closeModal() {
this.store.chosenModalTrainId = undefined;
this.tooltipStore.hide();
}
}
});
+3 -3
View File
@@ -75,18 +75,18 @@ export default defineComponent({
return positionString.charAt(0).toUpperCase() + positionString.slice(1); return positionString.charAt(0).toUpperCase() + positionString.slice(1);
}, },
displayStopList(stops: TrainStop[]): string | undefined { getTrainStopsHtml(stops: TrainStop[]): string {
if (!stops) return ''; if (!stops) return '';
return stops return stops
.reduce((acc: string[], stop: TrainStop, i: number) => { .reduce((acc: string[], stop: TrainStop, i: number) => {
if (stop.stopType.includes('ph') && !stop.stopNameRAW.includes('po.')) if (stop.stopType.includes('ph'))
acc.push( acc.push(
`<strong style='color:${stop.confirmed ? 'springgreen' : 'white'}'>${ `<strong style='color:${stop.confirmed ? 'springgreen' : 'white'}'>${
stop.stopName stop.stopName
}</strong>` }</strong>`
); );
else if (i > 0 && i < stops.length - 1 && !/po\.|sbl/gi.test(stop.stopNameRAW)) else if (i > 0 && i < stops.length - 1 && !/(, po$|sbl)/gi.test(stop.stopNameRAW))
acc.push( acc.push(
`<span style='color:${stop.confirmed ? 'springgreen' : 'lightgray'}'>${ `<span style='color:${stop.confirmed ? 'springgreen' : 'lightgray'}'>${
stop.stopName stop.stopName
+15 -1
View File
@@ -20,6 +20,15 @@ const routes: Array<RouteRecordRaw> = [
region: route.query.region region: route.query.region
}) })
}, },
{
path: '/driver',
name: 'DriverView',
component: () => import('../views/DriverView.vue'),
props: (route) => ({
trainId: route.query.trainId,
modalId: route.query.modalId
})
},
{ {
path: '/scenery', path: '/scenery',
name: 'SceneryView', name: 'SceneryView',
@@ -57,7 +66,12 @@ const routes: Array<RouteRecordRaw> = [
const router = createRouter({ const router = createRouter({
scrollBehavior(to, from, savedPosition) { scrollBehavior(to, from, savedPosition) {
if (to.name == 'SceneryView' && from.name !== to.name && from.query['view'] === undefined) if (
(to.name == 'SceneryView' || to.name == 'DriverView') &&
from.name !== to.name &&
from.query['view'] === undefined &&
!savedPosition
)
return { el: `.app_main`, top: -15 }; return { el: `.app_main`, top: -15 };
if (savedPosition) return savedPosition; if (savedPosition) return savedPosition;
+16 -27
View File
@@ -19,6 +19,7 @@ export const useApiStore = defineStore('apiStore', {
sceneryData: [] as StationJSONData[], sceneryData: [] as StationJSONData[],
nextUpdateTime: 0, nextUpdateTime: 0,
nextDataCheckTime: 0,
client: undefined as AxiosInstance | undefined, client: undefined as AxiosInstance | undefined,
@@ -48,17 +49,26 @@ export const useApiStore = defineStore('apiStore', {
}, },
async connectToAPI() { async connectToAPI() {
// Static data
this.fetchDonatorsData();
this.fetchStationsGeneralInfo();
this.fetchVehiclesInfo();
window.requestAnimationFrame(this.updateTick); window.requestAnimationFrame(this.updateTick);
}, },
updateTick(t: number) { updateTick(t: number) {
if (this.dataStatuses.connection == Status.Data.Offline) return; if (this.dataStatuses.connection == Status.Data.Offline) return;
// Static data refresh
if (t >= this.nextDataCheckTime) {
this.fetchDonatorsData();
this.fetchVehiclesInfo();
// Revalidation after staling
this.fetchStationsGeneralInfo().then(() => {
this.fetchStationsGeneralInfo();
});
this.nextDataCheckTime = t + 3600000;
}
// Active data fefresh
if (t >= this.nextUpdateTime) { if (t >= this.nextUpdateTime) {
this.fetchActiveData(); this.fetchActiveData();
this.nextUpdateTime = t + 20000; this.nextUpdateTime = t + 20000;
@@ -68,17 +78,6 @@ export const useApiStore = defineStore('apiStore', {
}, },
async fetchActiveData() { async fetchActiveData() {
// if (import.meta.env.VITE_API_ACTIVE_DATA_MODE == 'mocking') {
// import('../../tests/data/getActiveData.json').then((data) => {
// console.warn('activeData: mocking mode');
// this.activeData = data.default as API.ActiveData.Response;
// this.dataStatuses.connection = Status.Data.Loaded;
// });
// return;
// }
if (!this.activeData) this.dataStatuses.connection = Status.Data.Loading; if (!this.activeData) this.dataStatuses.connection = Status.Data.Loading;
try { try {
@@ -105,7 +104,7 @@ export const useApiStore = defineStore('apiStore', {
async fetchStationsGeneralInfo() { async fetchStationsGeneralInfo() {
try { try {
const sceneryData: StationJSONData[] = ( const sceneryData: StationJSONData[] = (
await this.client!.get<StationJSONData[]>('api/getSceneries') await this.client!.get<StationJSONData[]>(`api/getSceneries`)
).data; ).data;
this.dataStatuses.sceneries = Status.Data.Loaded; this.dataStatuses.sceneries = Status.Data.Loaded;
@@ -117,16 +116,6 @@ export const useApiStore = defineStore('apiStore', {
}, },
async fetchVehiclesInfo() { async fetchVehiclesInfo() {
// if (import.meta.env.VITE_API_VEHICLES_MODE == 'mocking') {
// import('../../tests/data/vehicles.json').then((data) => {
// console.warn('vehicles.json: mocking mode');
// this.vehiclesData = data.default;
// this.dataStatuses.vehicles = Status.Data.Loaded;
// });
// return;
// }
try { try {
const response = await this.client!.get<API.Vehicles.Response>('api/getVehicles'); const response = await this.client!.get<API.Vehicles.Response>('api/getVehicles');
+16 -14
View File
@@ -50,15 +50,6 @@ export const useMainStore = defineStore('mainStore', {
const timetable = train.timetable; const timetable = train.timetable;
const sceneryNames =
train.timetable?.sceneries?.map(
(sceneryHash) =>
apiStore.activeData?.activeSceneries?.find((st) => st.stationHash === sceneryHash)
?.stationName ??
apiStore.sceneryData.find((sd) => sd.hash === sceneryHash)?.name ??
sceneryHash
) ?? [];
const trainObj = { const trainObj = {
id: train.id, id: train.id,
modalId: `${train.driverName}${train.trainNo}`, // simplified id for train modal modalId: `${train.driverName}${train.trainNo}`, // simplified id for train modal
@@ -86,6 +77,13 @@ export const useMainStore = defineStore('mainStore', {
isSupporter: train.driverIsSupporter, isSupporter: train.driverIsSupporter,
driverLevel: train.driverLevel, driverLevel: train.driverLevel,
driverRouteLocation: {
name: 'DriverView',
query: {
trainId: train.id
}
},
timetableData: timetable timetableData: timetable
? { ? {
timetableId: timetable.timetableId, timetableId: timetable.timetableId,
@@ -96,7 +94,8 @@ export const useMainStore = defineStore('mainStore', {
followingStops: timetable.stopList, followingStops: timetable.stopList,
routeDistance: timetable.stopList[timetable.stopList.length - 1].stopDistance, routeDistance: timetable.stopList[timetable.stopList.length - 1].stopDistance,
sceneries: timetable.sceneries, sceneries: timetable.sceneries,
sceneryNames: sceneryNames.reverse(), warningNotes: timetable.warningNotes,
timetablePath: timetable.path.split(';').map((pathElementString) => { timetablePath: timetable.path.split(';').map((pathElementString) => {
const [arrival, station, departure] = pathElementString.split(','); const [arrival, station, departure] = pathElementString.split(',');
@@ -169,12 +168,15 @@ export const useMainStore = defineStore('mainStore', {
const offlineActiveSceneries = this.trainList.reduce((acc, train) => { const offlineActiveSceneries = this.trainList.reduce((acc, train) => {
if (!train.timetableData) return acc; if (!train.timetableData) return acc;
train.timetableData.sceneryNames.forEach((name) => { train.timetableData.timetablePath.forEach((p) => {
if ( if (
acc.findIndex((v) => v.name == name && v.region == train.region) != -1 || acc.findIndex(
(v) =>
(v.name == p.stationName || v.hash == p.stationHash) && v.region == train.region
) != -1 ||
apiStore.activeData?.activeSceneries?.findIndex( apiStore.activeData?.activeSceneries?.findIndex(
(sc) => (sc) =>
sc.stationName === name && (sc.stationName == p.stationName || sc.stationHash == p.stationHash) &&
sc.region == train.region && sc.region == train.region &&
Date.now() - sc.lastSeen < 1000 * 60 * 2 Date.now() - sc.lastSeen < 1000 * 60 * 2
) != -1 ) != -1
@@ -182,7 +184,7 @@ export const useMainStore = defineStore('mainStore', {
return acc; return acc;
acc.push({ acc.push({
name: name, name: p.stationName,
hash: '', hash: '',
region: train.region, region: train.region,
maxUsers: 0, maxUsers: 0,
+11 -9
View File
@@ -1,10 +1,18 @@
@import 'responsive.scss'; @import 'responsive.scss';
@import 'animations.scss'; @import 'animations.scss';
.journal-list {
display: flex;
flex-direction: column;
gap: 0.5em;
text-align: left;
margin-bottom: 0.5em;
}
.list_wrapper { .list_wrapper {
overflow-y: auto; overflow-y: auto;
height: 90vh; height: calc(100vh - 12.5em);
min-height: 650px; min-height: 500px;
margin-top: 0.5em; margin-top: 0.5em;
position: relative; position: relative;
@@ -12,7 +20,7 @@
} }
.journal_wrapper { .journal_wrapper {
max-width: 1500px; max-width: var(--max-container-width);
width: 100%; width: 100%;
margin: 0 auto; margin: 0 auto;
@@ -38,16 +46,10 @@
} }
} }
.journal_item {
cursor: pointer;
}
.journal_item, .journal_item,
.journal_warning { .journal_warning {
background-color: #1a1a1a; background-color: #1a1a1a;
padding: 1em; padding: 1em;
margin-bottom: 1em;
cursor: pointer;
} }
.journal_top-bar { .journal_top-bar {
+2 -18
View File
@@ -1,5 +1,6 @@
@import 'variables.scss'; @import 'variables.scss';
@import 'responsive.scss'; @import 'responsive.scss';
@import 'badge.scss';
.stats-tab { .stats-tab {
position: absolute; position: absolute;
@@ -7,7 +8,6 @@
z-index: 99; z-index: 99;
transform: translateY(1em); transform: translateY(1em);
width: 100%; width: 100%;
background-color: #1a1a1a; background-color: #1a1a1a;
@@ -29,26 +29,10 @@ hr.section-separator {
.info-stats { .info-stats {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
justify-content: center;
gap: 0.5em; gap: 0.5em;
} }
.stat-badge {
display: flex;
span {
background-color: $accentCol;
color: black;
font-weight: bold;
padding: 0.2em 0.5em;
}
span:first-child {
background-color: #333;
color: white;
}
}
@include smallScreen { @include smallScreen {
.journal-stats { .journal-stats {
text-align: center; text-align: center;
+2 -2
View File
@@ -41,11 +41,11 @@ $animType: ease-in-out;
} }
&-enter-active { &-enter-active {
transition: all $animDuration ease-out; transition: all $animDuration ease-in-out;
} }
&-leave-active { &-leave-active {
transition: all $animDuration ease-out; transition: all $animDuration ease-in-out;
} }
} }
+14 -3
View File
@@ -1,3 +1,6 @@
@import 'variables.scss';
@import 'responsive.scss';
.badge { .badge {
font-weight: 600; font-weight: 600;
@@ -78,14 +81,12 @@
.train-badge { .train-badge {
display: flex; display: flex;
align-items: center; align-items: center;
gap: 0.5em; gap: 0.3em;
padding: 0.1em 0.3em; padding: 0.1em 0.3em;
border-radius: 0.2em; border-radius: 0.2em;
font-weight: bold; font-weight: bold;
font-size: 0.9em;
&.twr { &.twr {
background-color: var(--clr-twr); background-color: var(--clr-twr);
box-shadow: 0 0 5px 1px var(--clr-twr); box-shadow: 0 0 5px 1px var(--clr-twr);
@@ -114,3 +115,13 @@
background-color: #007599; background-color: #007599;
} }
} }
.stat-badge {
margin: 0;
color: white;
& > span:first-child {
background-color: $accentCol;
color: black;
}
}
+28 -4
View File
@@ -124,10 +124,12 @@ input {
} }
a { a {
display: inline-block;
color: white;
text-decoration: none; text-decoration: none;
color: inherit;
}
a:not(.a-block):not(.a-button):not(.a-row) {
display: inline-block;
transition: color 0.3s; transition: color 0.3s;
@@ -138,6 +140,14 @@ a {
} }
} }
a.a-block {
display: block;
}
a.a-row {
display: table-row;
}
ul { ul {
padding: 0; padding: 0;
list-style: none; list-style: none;
@@ -184,6 +194,7 @@ a.a-button {
color: white; color: white;
background: none; background: none;
border-radius: 0.25em; border-radius: 0.25em;
text-decoration: none;
display: flex; display: flex;
align-items: center; align-items: center;
@@ -214,7 +225,11 @@ a.a-button {
font-weight: bold; font-weight: bold;
&:hover { &:hover {
background-color: #555; background-color: #424242;
}
&:disabled {
opacity: 0.75;
} }
} }
@@ -290,6 +305,7 @@ a.a-button {
// Basic tooltip // Basic tooltip
[data-tooltip] { [data-tooltip] {
cursor: help; cursor: help;
line-height: initial;
} }
[data-tooltip]:hover::after, [data-tooltip]:hover::after,
@@ -333,3 +349,11 @@ a.a-button {
width: 100%; width: 100%;
} }
} }
.g-separator {
display: block;
width: 100%;
height: 2px;
background-color: #aaa;
margin: 0.5em 0;
}
+17 -10
View File
@@ -4,6 +4,12 @@ export enum APIDataStatus {
OK = 'OK', OK = 'OK',
WARNING = 'WARNING' WARNING = 'WARNING'
} }
export interface APICache {
lastModified: string;
name: string;
}
export namespace API { export namespace API {
export namespace ActiveData { export namespace ActiveData {
export interface APIStatuses { export interface APIStatuses {
@@ -17,6 +23,7 @@ export namespace API {
activeSceneries?: API.ActiveSceneries.Response; activeSceneries?: API.ActiveSceneries.Response;
trains?: API.ActiveTrains.Response; trains?: API.ActiveTrains.Response;
apiStatuses?: APIStatuses; apiStatuses?: APIStatuses;
caches: APICache[];
} }
} }
@@ -197,6 +204,7 @@ export namespace API {
sceneries: string[]; sceneries: string[];
path: string; path: string;
warningNotes: string | null;
} }
} }
@@ -239,8 +247,6 @@ export namespace API {
authorName?: string; authorName?: string;
authorId?: number; authorId?: number;
stopsString?: string;
stockString?: string; stockString?: string;
stockHistory: string[]; stockHistory: string[];
@@ -248,17 +254,18 @@ export namespace API {
stockLength?: number; stockLength?: number;
maxSpeed?: number; maxSpeed?: number;
hashesString?: string;
currentSceneryName?: string; currentSceneryName?: string;
currentSceneryHash?: string; currentSceneryHash?: string;
routeSceneries?: string; routeSceneries: string;
checkpointArrivals?: string[]; checkpointArrivals: string[];
checkpointDepartures?: string[]; checkpointDepartures: string[];
checkpointArrivalsScheduled?: string[]; checkpointArrivalsScheduled: string[];
checkpointDeparturesScheduled?: string[]; checkpointDeparturesScheduled: string[];
checkpointStopTypes?: string[]; checkpointStopTypes: string[];
visitedSceneries?: string[]; visitedSceneries: string[];
sceneryNames: string[];
path: string; path: string;
warningNotes: string | null;
} }
export type Response = Data[]; export type Response = Data[];
+18 -12
View File
@@ -1,3 +1,5 @@
import { RouteLocationRaw } from 'vue-router';
export type Availability = 'default' | 'unavailable' | 'nonPublic' | 'abandoned' | 'nonDefault'; export type Availability = 'default' | 'unavailable' | 'nonPublic' | 'abandoned' | 'nonDefault';
export type ScenerySpawnType = 'passenger' | 'freight' | 'loco' | 'all'; export type ScenerySpawnType = 'passenger' | 'freight' | 'loco' | 'all';
@@ -70,18 +72,22 @@ export interface Train {
isTimeout: boolean; isTimeout: boolean;
isSupporter: boolean; isSupporter: boolean;
timetableData?: { driverRouteLocation: RouteLocationRaw;
timetableId: number;
category: string; timetableData?: TrainTimetableData;
route: string; }
followingStops: TrainStop[];
TWR: boolean; export interface TrainTimetableData {
SKR: boolean; timetableId: number;
routeDistance: number; category: string;
sceneries: string[]; route: string;
sceneryNames: string[]; followingStops: TrainStop[];
timetablePath: TimetablePathElement[]; TWR: boolean;
}; SKR: boolean;
routeDistance: number;
sceneries: string[];
timetablePath: TimetablePathElement[];
warningNotes: string | null;
} }
export interface Station { export interface Station {
+166
View File
@@ -0,0 +1,166 @@
<template>
<section class="driver-view">
<div class="view-wrapper">
<div v-if="chosenTrain">
<div class="actions">
<a class="a-button btn--image" @click="$router.back()">
<img src="/images/icon-back.svg" alt="train icon" />
<span>
{{ $t('trains.driver-return-link') }}
</span>
</a>
<router-link
:to="`/journal/timetables?search-driver=${chosenTrain.driverName}`"
class="a-button btn--image"
>
<span class="hidable">
{{ $t('trains.driver-journal-link') }}
</span>
<img src="/images/icon-train.svg" alt="train icon" />
</router-link>
</div>
<div class="train-card">
<TrainInfo :train="chosenTrain" :extended="true" ref="trainInfo" />
<TrainSchedule :train="chosenTrain" />
</div>
</div>
<Loading v-else-if="apiStore.dataStatuses.connection == Status.Data.Loading" />
<div v-else class="driver-not-found">
<h2>&olcross; {{ $t('trains.driver-not-found-header') }}</h2>
<p class="text--grayed">
{{ $t('trains.driver-not-found-desc-1') }} <br />
{{ $t('trains.driver-not-found-desc-2') }}
<router-link to="/journal/timetables"
>{{ $t('trains.driver-not-found-journal') }} </router-link
>!
</p>
<p v-if="props.trainId && otherDriverTrains.length > 0">
<i18n-t keypath="trains.driver-not-found-others">
<template v-slot:driver>
<b>{{ otherDriverTrains[0].driverName }}</b>
</template>
</i18n-t>
</p>
<div class="other-driver-trains">
<template v-for="(train, i) in otherDriverTrains">
<router-link :to="`/driver?trainId=${train.id}`">
{{ train.trainNo }}
| {{ regionsJSON.find((r) => r.id == train.region)?.name ?? 'PL1' }}
</router-link>
</template>
</div>
<div style="margin-top: 1em">
<router-link to="/">&lt;&lt; {{ $t('trains.driver-not-found-return') }}</router-link>
</div>
</div>
</div>
</section>
</template>
<script setup lang="ts">
import { computed } from 'vue';
import TrainInfo from '../components/TrainsView/TrainInfo.vue';
import TrainSchedule from '../components/TrainsView/TrainSchedule.vue';
import Loading from '../components/Global/Loading.vue';
import { useMainStore } from '../store/mainStore';
import { useApiStore } from '../store/apiStore';
import { Status } from '../typings/common';
import { regions as regionsJSON } from '../data/options.json';
const props = defineProps({
trainId: {
type: String
},
modalId: {
type: String
}
});
const mainStore = useMainStore();
const apiStore = useApiStore();
const chosenTrain = computed(() =>
mainStore.trainList.find((train) => train.id == props.trainId || train.modalId == props.modalId)
);
const otherDriverTrains = computed(() => {
return mainStore.trainList.filter(
(train) =>
train.driverId == Number(props.trainId?.split('|')[0]) &&
(train.timetableData || train.online || train.lastSeen >= Date.now() - 60000)
);
});
</script>
<style lang="scss" scoped>
@import '../styles/responsive';
$viewBgCol: #1a1a1a;
.driver-view {
margin: 0 auto;
padding: 1em 0;
max-width: var(--max-container-width);
min-height: calc(100vh - 7em);
}
.actions {
display: flex;
align-items: flex-end;
justify-content: space-between;
gap: 0.5em;
}
.actions > a {
background-color: $viewBgCol;
padding: 0.5em;
border-radius: 0.5em 0.5em 0 0;
&:hover {
background-color: lighten($viewBgCol, 10);
}
}
.train-card {
background-color: $viewBgCol;
border-radius: 0 0 0.5em 0.5em;
}
.driver-not-found {
background-color: $viewBgCol;
text-align: center;
padding: 1em;
border-radius: 0.5em 0.5em;
p {
padding: 0.5em 0;
}
a {
text-decoration: underline;
color: white;
}
}
.other-driver-trains {
display: flex;
justify-content: center;
flex-wrap: wrap;
gap: 0.5em;
}
@include smallScreen {
.actions > a > span.hidable {
display: none;
}
}
</style>
+2 -11
View File
@@ -40,7 +40,6 @@ import { defineComponent, provide, reactive, Ref, ref } from 'vue';
import dateMixin from '../mixins/dateMixin'; import dateMixin from '../mixins/dateMixin';
import routerMixin from '../mixins/routerMixin'; import routerMixin from '../mixins/routerMixin';
import modalTrainMixin from '../mixins/modalTrainMixin';
import JournalOptions from '../components/JournalView/JournalOptions.vue'; import JournalOptions from '../components/JournalView/JournalOptions.vue';
import JournalStats from '../components/JournalView/JournalStats.vue'; import JournalStats from '../components/JournalView/JournalStats.vue';
@@ -148,7 +147,7 @@ export default defineComponent({
JournalHeader, JournalHeader,
JournalTimetablesList JournalTimetablesList
}, },
mixins: [dateMixin, routerMixin, modalTrainMixin], mixins: [dateMixin, routerMixin],
name: 'JournalTimetables', name: 'JournalTimetables',
@@ -307,14 +306,6 @@ export default defineComponent({
this.searchersValues[v as Journal.TimetableSearchKey] = options[v] ?? ''; this.searchersValues[v as Journal.TimetableSearchKey] = options[v] ?? '';
}); });
// this.searchersValues['search-date'] = options['search-date'] ?? '';
// this.searchersValues['search-driver'] = options['search-driver'] ?? '';
// this.searchersValues['search-train'] = options['search-train'] ?? '';
// this.searchersValues['search-dispatcher'] = options['search-dispatcher'] ?? '';
// this.searchersValues['search-issuedFrom'] = options['search-issuedFrom'] ?? '';
// this.searchersValues['search-via'] = options['search-via'] ?? '';
// this.searchersValues['search-terminatingAt'] = options['search-terminatingAt'] ?? '';
this.sorterActive.id = this.sorterActive.id =
(options['sorter-active'] as Journal.TimetableSorterKey) ?? 'timetableId'; (options['sorter-active'] as Journal.TimetableSorterKey) ?? 'timetableId';
@@ -462,7 +453,7 @@ export default defineComponent({
this.timetableHistory = responseData; this.timetableHistory = responseData;
// Stats display // Stats display
this.store.driverStatsName = this.mainStore.driverStatsName =
this.timetableHistory.length > 0 && this.searchersValues['search-driver'].trim() this.timetableHistory.length > 0 && this.searchersValues['search-driver'].trim()
? this.timetableHistory[0].driverName ? this.timetableHistory[0].driverName
: ''; : '';
+3 -1
View File
@@ -13,6 +13,7 @@
:station="stationInfo" :station="stationInfo"
:onlineScenery="onlineSceneryInfo" :onlineScenery="onlineSceneryInfo"
/> />
<SceneryInfo :station="stationInfo" :onlineScenery="onlineSceneryInfo" /> <SceneryInfo :station="stationInfo" :onlineScenery="onlineSceneryInfo" />
</div> </div>
@@ -229,10 +230,11 @@ button.back-btn {
overflow: auto; overflow: auto;
background-color: #181818; background-color: #181818;
border-radius: 0.5em;
padding: 1em 0.5em; padding: 1em 0.5em;
height: calc(100vh - 0.5em); height: calc(100vh - 0.5em);
min-height: 800px; min-height: 500px;
max-height: 2000px; max-height: 2000px;
} }
+9 -2
View File
@@ -8,6 +8,8 @@
ref="filterCardRef" ref="filterCardRef"
/> />
<StationStats />
<button <button
class="btn-donation btn--image" class="btn-donation btn--image"
ref="btn" ref="btn"
@@ -21,7 +23,6 @@
<DonationCard :is-card-open="isDonationCardOpen" @toggle-card="toggleDonationCard" /> <DonationCard :is-card-open="isDonationCardOpen" @toggle-card="toggleDonationCard" />
<StationTable @toggle-donation-card="toggleDonationCard" /> <StationTable @toggle-donation-card="toggleDonationCard" />
<StationStats />
</div> </div>
</section> </section>
</template> </template>
@@ -96,14 +97,16 @@ export default defineComponent({
.stations-options { .stations-options {
display: flex; display: flex;
justify-content: space-between;
flex-wrap: wrap; flex-wrap: wrap;
gap: 0.5em; gap: 0.5em;
position: relative;
margin-bottom: 0.5em; margin-bottom: 0.5em;
} }
button.btn-donation { button.btn-donation {
margin-left: auto;
$btnColor: #254069; $btnColor: #254069;
background-color: $btnColor; background-color: $btnColor;
@@ -118,4 +121,8 @@ button.btn-donation {
} }
} }
} }
.count {
padding: 0.5em;
}
</style> </style>
+7 -11
View File
@@ -19,7 +19,6 @@
import { computed, ComputedRef, defineComponent, provide, reactive, ref, watch } from 'vue'; import { computed, ComputedRef, defineComponent, provide, reactive, ref, watch } from 'vue';
import TrainOptions from '../components/TrainsView/TrainOptions.vue'; import TrainOptions from '../components/TrainsView/TrainOptions.vue';
import TrainTable from '../components/TrainsView/TrainTable.vue'; import TrainTable from '../components/TrainsView/TrainTable.vue';
import modalTrainMixin from '../mixins/modalTrainMixin';
import { useMainStore } from '../store/mainStore'; import { useMainStore } from '../store/mainStore';
import { TrainFilter, trainFilters } from '../components/TrainsView/typings'; import { TrainFilter, trainFilters } from '../components/TrainsView/typings';
import { filteredTrainList } from '../managers/trainFilterManager'; import { filteredTrainList } from '../managers/trainFilterManager';
@@ -33,8 +32,6 @@ export default defineComponent({
TrainStats TrainStats
}, },
mixins: [modalTrainMixin],
props: { props: {
train: { train: {
type: String, type: String,
@@ -102,16 +99,16 @@ export default defineComponent({
}, },
activated() { activated() {
// Backwards compatibility with external links leading to train modal
if (this.trainId) {
this.$router.replace(`/driver?modalId=${this.trainId}`);
return;
}
if (this.train) { if (this.train) {
this.searchedTrain = this.train; this.searchedTrain = this.train;
this.searchedDriver = this.driver || ''; this.searchedDriver = this.driver || '';
} }
this.$nextTick(() => {
if (this.trainId) {
this.selectModalTrainById(this.trainId);
}
});
} }
}); });
</script> </script>
@@ -120,13 +117,12 @@ export default defineComponent({
@import '../styles/responsive.scss'; @import '../styles/responsive.scss';
.trains-view { .trains-view {
min-height: 600px;
position: relative; position: relative;
} }
.trains_wrapper { .trains_wrapper {
margin: 1rem auto; margin: 1rem auto;
max-width: 1500px; max-width: var(--max-container-width);
} }
.trains_topbar { .trains_topbar {