Ładowanie danych nt pociągów i rozkładów jazdy z API Stacjownika

This commit is contained in:
2022-03-07 22:59:10 +01:00
parent cd05fc74cf
commit 6296f1e89a
22 changed files with 494 additions and 644 deletions
+13 -13
View File
@@ -50,9 +50,9 @@
</span> </span>
<img src="@/assets/icon-dispatcher.svg" alt="icon dispatcher" /> <img src="@/assets/icon-dispatcher.svg" alt="icon dispatcher" />
<span class="text--primary">{{ data.activeStationCount }}</span> <span class="text--primary">{{ dispatcherCount }}</span>
<span class="text--grayed">|</span> <span class="text--grayed">|</span>
<span class="text--primary">{{ data.activeTrainCount }}</span> <span class="text--primary">{{ trainCount }}</span>
<img src="@/assets/icon-train.svg" alt="icon train" /> <img src="@/assets/icon-train.svg" alt="icon train" />
</div> </div>
</span> </span>
@@ -70,9 +70,9 @@
<main class="app_main"> <main class="app_main">
<router-view v-slot="{ Component }"> <router-view v-slot="{ Component }">
<!-- <transition name="view-anim" mode="out-in"> --> <!-- <transition name="view-anim" mode="out-in"> -->
<keep-alive> <keep-alive>
<component :is="Component" /> <component :is="Component" />
</keep-alive> </keep-alive>
<!-- </transition> --> <!-- </transition> -->
</router-view> </router-view>
</main> </main>
@@ -92,7 +92,7 @@
import Clock from '@/components/App/Clock.vue'; import Clock from '@/components/App/Clock.vue';
import StorageManager from '@/scripts/managers/storageManager'; import StorageManager from '@/scripts/managers/storageManager';
import { computed, ComputedRef, defineComponent, provide, ref } from 'vue'; import { computed, ComputedRef, defineComponent, provide, reactive, Ref, ref } from 'vue';
import { GETTERS } from './constants/storeConstants'; import { GETTERS } from './constants/storeConstants';
import { StoreData } from './scripts/interfaces/StoreData'; import { StoreData } from './scripts/interfaces/StoreData';
import { useStore } from './store'; import { useStore } from './store';
@@ -116,8 +116,8 @@ export default defineComponent({
() => store.getters[GETTERS.currentRegion] () => store.getters[GETTERS.currentRegion]
); );
const dataStatus = computed(() => data.value); const dataStatus = computed(() => data.value.sceneryDataStatus);
const sceneryDataStatus = computed(() => data.value.sceneryDataStatus); // const sceneryDataStatus = computed(() => data.value.sceneryDataStatus);
const isFilterCardVisible = ref(false); const isFilterCardVisible = ref(false);
@@ -129,13 +129,13 @@ export default defineComponent({
isFilterCardVisible, isFilterCardVisible,
dataStatus, dataStatus,
sceneryDataStatus,
dispatcherDataStatus: computed(() => data.value.dispatcherDataStatus), dispatcherDataStatus: computed(() => data.value.dispatcherDataStatus),
timetableCount: data.value.trainList.filter((train) => train.timetableData).length, trainCount: computed(() => data.value.trainList.length),
onlineDispatcherCount: data.value.stationList.filter(
(station) => station.onlineInfo && station.onlineInfo.statusTimestamp > 0 dispatcherCount: computed(
).length, () => data.value.stationList.filter((station) => station.onlineInfo).length
),
openFilterCard() { openFilterCard() {
isFilterCardVisible.value = true; isFilterCardVisible.value = true;
-23
View File
@@ -195,18 +195,9 @@ export default defineComponent({
watch: { watch: {
dataStatus(storeData: StoreData) { dataStatus(storeData: StoreData) {
const dataConnectionStatus = storeData.dataConnectionStatus;
const sceneryDataStatus = storeData.sceneryDataStatus; const sceneryDataStatus = storeData.sceneryDataStatus;
const trainsDataStatus = storeData.trainsDataStatus; const trainsDataStatus = storeData.trainsDataStatus;
const dispatcherDataStatus = storeData.dispatcherDataStatus; const dispatcherDataStatus = storeData.dispatcherDataStatus;
const timetableDataStatus = storeData.timetableDataStatus;
if (dataConnectionStatus == DataStatus.Error) {
this.setSignalStatus(dataConnectionStatus);
this.indicator.status = dataConnectionStatus;
this.indicator.message = 'data-status.S1a-connection';
return;
}
if (sceneryDataStatus == DataStatus.Error) { if (sceneryDataStatus == DataStatus.Error) {
this.setSignalStatus(sceneryDataStatus); this.setSignalStatus(sceneryDataStatus);
@@ -229,20 +220,6 @@ export default defineComponent({
return; return;
} }
if (timetableDataStatus == DataStatus.Warning) {
this.setSignalStatus(timetableDataStatus);
this.indicator.status = timetableDataStatus;
this.indicator.message = 'data-status.S5-timetables';
return;
}
if (timetableDataStatus == DataStatus.Loading) {
this.setSignalStatus(timetableDataStatus);
this.indicator.status = timetableDataStatus;
this.indicator.message = 'data-status.S3';
return;
}
this.setSignalStatus(DataStatus.Loaded); this.setSignalStatus(DataStatus.Loaded);
this.indicator.status = DataStatus.Loaded; this.indicator.status = DataStatus.Loaded;
+119
View File
@@ -0,0 +1,119 @@
<template>
<span class="stop-date">
<span
class="date arrival"
v-if="!stop.beginsHere"
:class="{
delayed: stop.arrivalDelay > 0 && stop.confirmed,
preponed: stop.arrivalDelay < 0 && stop.confirmed,
'on-time': stop.arrivalDelay == 0 && stop.confirmed,
}"
>
<span v-if="stop.arrivalDelay != 0 && stop.confirmed">
<s>{{ timestampToString(stop.arrivalTimestamp) }}</s>
{{ timestampToString(stop.arrivalRealTimestamp) }}
({{ stop.arrivalDelay > 0 ? '+' : '' }}{{ stop.arrivalDelay }})
</span>
<span v-else>
{{ timestampToString(stop.arrivalTimestamp) }}
</span>
</span>
<span class="date stop" v-if="stop.stopTime" :class="stop.stopType.replace(', ', '-')">
{{ stop.stopTime }} {{ stop.stopType == '' ? 'pt' : stop.stopType }}
</span>
<span
class="date departure"
v-if="!stop.terminatesHere && stop.stopTime != 0"
:class="{
delayed: stop.departureDelay > 0 && stop.confirmed,
preponed: stop.departureDelay < 0 && stop.confirmed,
}"
>
<span v-if="stop.departureDelay != 0 && stop.confirmed">
<s>{{ timestampToString(stop.departureTimestamp) }}</s>
{{ timestampToString(stop.departureRealTimestamp) }}
({{ stop.departureDelay > 0 ? '+' : '' }}{{ stop.departureDelay }})
</span>
<span v-else>
{{ timestampToString(stop.departureTimestamp) }}
</span>
</span>
</span>
</template>
<script lang="ts">
import dateMixin from '@/mixins/dateMixin';
import TrainStop from '@/scripts/interfaces/TrainStop';
import { defineComponent } from 'vue';
export default defineComponent({
mixins: [dateMixin],
props: {
stop: {
type: Object as () => TrainStop,
required: true,
},
},
setup() {
return {};
},
});
</script>
<style lang="scss" scoped>
$preponedClr: lime;
$delayedClr: salmon;
$dateClr: #525151;
$stopExchangeClr: #db8e29;
$stopDefaultClr: #252525;
.stop-date {
display: flex;
align-items: center;
.date {
background: $dateClr;
padding: 0.3em 0.5em;
}
.stop {
&.ph,
&.ph-pm,
&.pm {
background: $stopExchangeClr;
}
background: $stopDefaultClr;
}
.arrival,
.departure {
&.delayed {
s {
color: #999;
}
span {
color: $delayedClr;
}
}
&.preponed {
s {
color: #999;
}
span {
color: $preponedClr;
}
}
}
}
</style>
@@ -17,7 +17,7 @@
<span class="status-badge" v-if="station.onlineInfo" :class="station.onlineInfo.statusID"> <span class="status-badge" v-if="station.onlineInfo" :class="station.onlineInfo.statusID">
{{ $t(`status.${station.onlineInfo.statusID}`) }} {{ $t(`status.${station.onlineInfo.statusID}`) }}
{{ station.onlineInfo.statusID == 'online' ? station.onlineInfo.statusTimeString : '' }} {{ station.onlineInfo.statusID == 'online' ? timestampToString(station.onlineInfo.statusTimestamp) : '' }}
</span> </span>
<span class="status-badge free" v-else> <span class="status-badge free" v-else>
@@ -31,9 +31,10 @@ import { defineComponent } from 'vue';
import styleMixin from '@/mixins/styleMixin'; import styleMixin from '@/mixins/styleMixin';
import Station from '@/scripts/interfaces/Station'; import Station from '@/scripts/interfaces/Station';
import dateMixin from '@/mixins/dateMixin';
export default defineComponent({ export default defineComponent({
mixins: [styleMixin], mixins: [styleMixin, dateMixin ],
props: { props: {
station: { station: {
type: Object as () => Station, type: Object as () => Station,
+44 -21
View File
@@ -21,8 +21,8 @@
</div> </div>
<transition name="scenery-timetable-list-anim" mode="out-in"> <transition name="scenery-timetable-list-anim" mode="out-in">
<div :key="timetableDataStatus + selectedCheckpoint"> <div :key="trainsDataStatus + selectedCheckpoint">
<span class="timetable-item loading" v-if="timetableDataStatus == 0 && computedScheduledTrains.length == 0"> <span class="timetable-item loading" v-if="trainsDataStatus == 0 && computedScheduledTrains.length == 0">
{{ $t('app.loading') }} {{ $t('app.loading') }}
</span> </span>
@@ -65,18 +65,26 @@
</span> </span>
<span class="timetable-schedule"> <span class="timetable-schedule">
<span class="schedule-arrival"> <span class="schedule-arrival">
<span <span class="arrival-time begins" v-if="scheduledTrain.stopInfo.beginsHere">
class="arrival-time begins" {{ $t('timetables.begins') }}
v-if="scheduledTrain.stopInfo.beginsHere"
v-html="$t('timetables.begins')"
>
</span> </span>
<span class="arrival-time" v-else> <span class="arrival-time" v-else>
{{ scheduledTrain.stopInfo.arrivalTimeString }} ({{ scheduledTrain.stopInfo.arrivalDelay }}) <div v-if="scheduledTrain.stopInfo.arrivalDelay == 0">
<span>{{ timestampToString(scheduledTrain.stopInfo.arrivalTimestamp) }}</span>
</div>
<div v-else>
<s style="margin-right: 0.2em;" class="text--grayed">{{
timestampToString(scheduledTrain.stopInfo.arrivalTimestamp)
}}</s>
<span>
{{ timestampToString(scheduledTrain.stopInfo.arrivalRealTimestamp) }}
({{ scheduledTrain.stopInfo.arrivalDelay > 0 ? '+' : ''
}}{{ scheduledTrain.stopInfo.arrivalDelay }})
</span>
</div>
</span> </span>
</span> </span>
<span class="schedule-stop"> <span class="schedule-stop">
@@ -88,15 +96,25 @@
</span> </span>
<span class="schedule-departure"> <span class="schedule-departure">
<span <span class="departure-time terminates" v-if="scheduledTrain.stopInfo.terminatesHere">
class="departure-time terminates" {{ $t('timetables.terminates') }}
v-if="scheduledTrain.stopInfo.terminatesHere"
v-html="$t('timetables.terminates')"
>
</span> </span>
<span class="departure-time" v-else> <span class="departure-time" v-else>
{{ scheduledTrain.stopInfo.departureTimeString }} ({{ scheduledTrain.stopInfo.departureDelay }}) <div v-if="scheduledTrain.stopInfo.departureDelay == 0">
<span>{{ timestampToString(scheduledTrain.stopInfo.departureTimestamp) }}</span>
</div>
<div v-else>
<s style="margin-right: 0.2em;" class="text--grayed">{{
timestampToString(scheduledTrain.stopInfo.departureTimestamp)
}}</s>
<span>
{{ timestampToString(scheduledTrain.stopInfo.departureRealTimestamp) }}
({{ scheduledTrain.stopInfo.departureDelay > 0 ? '+' : ''
}}{{ scheduledTrain.stopInfo.departureDelay }})
</span>
</div>
</span> </span>
</span> </span>
</span> </span>
@@ -115,10 +133,13 @@ import { useStore } from '@/store';
import { GETTERS } from '@/constants/storeConstants'; import { GETTERS } from '@/constants/storeConstants';
import { DataStatus } from '@/scripts/enums/DataStatus'; import { DataStatus } from '@/scripts/enums/DataStatus';
import { ComputedRef } from 'vue'; import { ComputedRef } from 'vue';
import dateMixin from '@/mixins/dateMixin';
export default defineComponent({ export default defineComponent({
components: { SelectBox }, components: { SelectBox },
mixins: [dateMixin],
props: { props: {
station: { station: {
type: Object as () => Station, type: Object as () => Station,
@@ -142,7 +163,7 @@ export default defineComponent({
const store = useStore(); const store = useStore();
const timetableDataStatus = computed(() => store.getters[GETTERS.timetableDataStatus]) as ComputedRef<DataStatus>; const trainsDataStatus = computed(() => store.getters[GETTERS.trainsDataStatus]) as ComputedRef<DataStatus>;
const selectedCheckpoint = ref( const selectedCheckpoint = ref(
props.station?.generalInfo?.checkpoints?.length == 0 props.station?.generalInfo?.checkpoints?.length == 0
@@ -176,7 +197,7 @@ export default defineComponent({
currentURL, currentURL,
selectedCheckpoint, selectedCheckpoint,
computedScheduledTrains, computedScheduledTrains,
timetableDataStatus, trainsDataStatus,
}; };
}, },
@@ -317,11 +338,12 @@ h3 {
&-schedule { &-schedule {
display: grid; display: grid;
grid-template-columns: repeat(auto-fit, minmax(30px, 1fr)); grid-template-columns: repeat(auto-fit, minmax(30px, 1fr));
font-size: 1.3em; font-size: 1.15em;
@include smallScreen() { @include smallScreen() {
width: 100%; width: 100%;
margin: 0.5em 0; margin: 0.5em 0;
font-size: 1.4em;
} }
} }
} }
@@ -378,7 +400,6 @@ h3 {
.general-info { .general-info {
.info-number { .info-number {
color: $accentCol; color: $accentCol;
} }
.info-route { .info-route {
@@ -396,6 +417,8 @@ h3 {
} }
.general-status { .general-status {
margin-left: 0.5em;
span.arriving { span.arriving {
color: #aaa; color: #aaa;
} }
@@ -449,6 +472,6 @@ h3 {
.arrival-time.begins, .arrival-time.begins,
.departure-time.terminates { .departure-time.terminates {
font-size: 0.75em; font-size: 0.85em;
} }
</style> </style>
+7 -5
View File
@@ -87,7 +87,7 @@
<td class="station_status"> <td class="station_status">
<span class="status-badge" :class="station.onlineInfo.statusID" v-if="station.onlineInfo"> <span class="status-badge" :class="station.onlineInfo.statusID" v-if="station.onlineInfo">
{{ $t(`status.${station.onlineInfo.statusID}`) }} {{ $t(`status.${station.onlineInfo.statusID}`) }}
{{ station.onlineInfo.statusID == 'online' ? station.onlineInfo.statusTimeString : '' }} {{ station.onlineInfo.statusID == 'online' ? timestampToString(station.onlineInfo.statusTimestamp) : '' }}
</span> </span>
<span class="status-badge free" v-else> <span class="status-badge free" v-else>
@@ -102,7 +102,7 @@
<td class="station_dispatcher-exp"> <td class="station_dispatcher-exp">
<span <span
v-if="station.onlineInfo" v-if="station.onlineInfo"
:style="calculateExpStyle(station.onlineInfo.dispatcherExp, station.onlineInfo.dispatcherIsSupporter)" :style="calculateExpStyle(station.onlineInfo.dispatcherExp)"
> >
{{ 2 > station.onlineInfo.dispatcherExp ? 'L' : station.onlineInfo.dispatcherExp }} {{ 2 > station.onlineInfo.dispatcherExp ? 'L' : station.onlineInfo.dispatcherExp }}
</span> </span>
@@ -231,6 +231,8 @@ import { computed, ComputedRef, defineComponent } from '@vue/runtime-core';
import { useStore } from '@/store'; import { useStore } from '@/store';
import { GETTERS } from '@/constants/storeConstants'; import { GETTERS } from '@/constants/storeConstants';
import Station from '@/scripts/interfaces/Station'; import Station from '@/scripts/interfaces/Station';
import { StoreData } from '@/scripts/interfaces/StoreData';
import dateMixin from '@/mixins/dateMixin';
export default defineComponent({ export default defineComponent({
props: { props: {
@@ -248,7 +250,7 @@ export default defineComponent({
changeSorter: { type: Function, required: true }, changeSorter: { type: Function, required: true },
}, },
mixins: [styleMixin], mixins: [styleMixin, dateMixin],
data: () => ({ data: () => ({
likeIcon: require('@/assets/icon-like.svg'), likeIcon: require('@/assets/icon-like.svg'),
@@ -275,10 +277,10 @@ export default defineComponent({
setup() { setup() {
const store = useStore(); const store = useStore();
const dataConnectionStatus: ComputedRef<DataStatus> = computed(() => store.getters[GETTERS.dataStatus]); const data: ComputedRef<StoreData> = computed(() => store.getters[GETTERS.allData]);
const isDataLoaded = computed(() => { const isDataLoaded = computed(() => {
return dataConnectionStatus.value == DataStatus.Loaded; return data.value.sceneryDataStatus == DataStatus.Loaded;
}); });
return { return {
+7 -94
View File
@@ -22,51 +22,8 @@
<span class="content" v-html="stop.comments"> </span> <span class="content" v-html="stop.comments"> </span>
</span> </span>
</span> </span>
<span class="stop-date">
<span
class="date arrival"
v-if="!stop.beginsHere"
:class="{
delayed: stop.arrivalDelay > 0 && stop.confirmed,
preponed: stop.arrivalDelay < 0 && stop.confirmed,
'on-time': stop.arrivalDelay == 0 && stop.confirmed,
}"
>
<span v-if="stop.arrivalDelay != 0 && stop.confirmed">
<s>{{ stop.arrivalTimeString }}</s>
{{ stop.arrivalRealTimeString }}
({{ stop.arrivalDelay > 0 ? '+' : '' }}{{ stop.arrivalDelay }})
</span>
<span v-else> <stop-date :stop="stop" />
{{ stop.arrivalTimeString }}
</span>
</span>
<span class="date stop" v-if="stop.stopTime" :class="stop.stopType.replace(', ', '-')">
{{ stop.stopTime }} {{ stop.stopType == '' ? 'pt' : stop.stopType }}
</span>
<span
class="date departure"
v-if="!stop.terminatesHere && stop.stopTime != 0"
:class="{
delayed: stop.departureDelay > 0 && stop.confirmed,
preponed: stop.departureDelay < 0 && stop.confirmed,
}"
>
<span v-if="stop.departureDelay != 0 && stop.confirmed">
<s>{{ stop.departureTimeString }}</s>
{{ stop.departureRealTimeString }}
({{ stop.departureDelay > 0 ? '+' : '' }}{{ stop.departureDelay }})
</span>
<span v-else>
{{ stop.departureTimeString }}
</span>
</span>
</span>
</span> </span>
<div class="stop_line" v-if="i < followingStops.length - 1"> <div class="stop_line" v-if="i < followingStops.length - 1">
@@ -88,10 +45,13 @@
</template> </template>
<script lang="ts"> <script lang="ts">
import TrainStop from '@/scripts/interfaces/TrainStop';
import { computed, defineComponent } from '@vue/runtime-core'; import { computed, defineComponent } from '@vue/runtime-core';
import dateMixin from '@/mixins/dateMixin';
import TrainStop from '@/scripts/interfaces/TrainStop';
import StopDate from '../Global/StopDate.vue';
export default defineComponent({ export default defineComponent({
components: { StopDate },
props: { props: {
followingStops: { followingStops: {
type: Array as () => TrainStop[], type: Array as () => TrainStop[],
@@ -99,6 +59,8 @@ export default defineComponent({
}, },
}, },
mixins: [dateMixin],
emits: ['click'], emits: ['click'],
data: () => ({ data: () => ({
@@ -159,12 +121,6 @@ $barClr: #b1b1b1;
$confirmedClr: #18d818; $confirmedClr: #18d818;
$stoppedClr: #f55f31; $stoppedClr: #f55f31;
$haltClr: #f8bb36; $haltClr: #f8bb36;
$preponedClr: lime;
$delayedClr: salmon;
$dateClr: #525151;
$stopExchangeClr: #db8e29;
$stopDefaultClr: #252525;
$stopNameClr: #22a8d1; $stopNameClr: #22a8d1;
@keyframes blink { @keyframes blink {
@@ -220,49 +176,6 @@ $stopNameClr: #22a8d1;
} }
} }
.stop-date {
display: flex;
align-items: center;
.date {
background: $dateClr;
padding: 0.3em 0.5em;
}
.stop {
&.ph,
&.ph-pm,
&.pm {
background: $stopExchangeClr;
}
background: $stopDefaultClr;
}
.arrival,
.departure {
&.delayed {
s {
color: #999;
}
span {
color: $delayedClr;
}
}
&.preponed {
s {
color: #999;
}
span {
color: $preponedClr;
}
}
}
}
ul.stop_list > li.stop { ul.stop_list > li.stop {
position: relative; position: relative;
-4
View File
@@ -80,7 +80,6 @@ export default defineComponent({
setup(props) { setup(props) {
const store = useStore(); const store = useStore();
const timetableDataStatus: ComputedRef<DataStatus> = computed(() => store.getters[GETTERS.timetableDataStatus]);
const trainsDataStatus: ComputedRef<DataStatus> = computed(() => store.getters[GETTERS.trainsDataStatus]); const trainsDataStatus: ComputedRef<DataStatus> = computed(() => store.getters[GETTERS.trainsDataStatus]);
const searchedTrain = inject('searchedTrain') as Ref<string>; const searchedTrain = inject('searchedTrain') as Ref<string>;
@@ -108,9 +107,6 @@ export default defineComponent({
sorterActive: inject('sorterActive') as { id: string | number; dir: number }, sorterActive: inject('sorterActive') as { id: string | number; dir: number },
trainsDataStatus: computed(() => trainsDataStatus.value), trainsDataStatus: computed(() => trainsDataStatus.value),
timetableLoaded: computed(() => timetableDataStatus.value === DataStatus.Loaded),
timetableWarning: computed(() => timetableDataStatus.value === DataStatus.Warning),
timetableError: computed(() => timetableDataStatus.value === DataStatus.Error),
distanceLimitExceeded: computed( distanceLimitExceeded: computed(
() => props.trains.findIndex(({ timetableData }) => timetableData && timetableData.routeDistance > 200) != -1 () => props.trains.findIndex(({ timetableData }) => timetableData && timetableData.routeDistance > 200) != -1
), ),
@@ -6,7 +6,7 @@
</div> </div>
<train-info :train="train" /> <train-info :train="train" />
<train-schedule :followingStops="train.timetableData?.followingStops" ref="card-inner" tabindex="0" /> <train-schedule v-if="train.timetableData" :followingStops="train.timetableData.followingStops" ref="card-inner" tabindex="0" />
</section> </section>
</template> </template>
+4 -10
View File
@@ -1,22 +1,18 @@
export const ACTIONS = { export const ACTIONS = {
synchronizeData: "synchronizeData", synchronizeData: "synchronizeData",
fetchOnlineData: "fetchOnlineData", fetchOnlineData: "fetchOnlineData",
fetchTimetableData: "fetchTimetableData" loadStaticStationData: "loadStaticStationData"
} }
export const MUTATIONS = { export const MUTATIONS = {
SET_SCENERY_DATA: "SET_SCENERY_DATA",
SET_DATA_CONNECTION_STATUS: "SET_DATA_CONNECTION_STATUS", SET_DATA_CONNECTION_STATUS: "SET_DATA_CONNECTION_STATUS",
SET_SCENERY_DATA: "SET_SCENERY_DATA",
SET_SCENERY_DATA_STATUS: "SET_SCENERY_DATA_STATUS", SET_SCENERY_DATA_STATUS: "SET_SCENERY_DATA_STATUS",
SET_TIMETABLE_DATA_STATUS: "SET_TIMETABLE_DATA_STATUS",
SET_DISPATCHER_DATA_STATUS: "SET_DISPATCHER_DATA_STATUS", SET_DISPATCHER_DATA_STATUS: "SET_DISPATCHER_DATA_STATUS",
SET_TRAINS_DATA_STATUS: "SET_TRAINS_DATA_STATUS", SET_TRAINS_DATA_STATUS: "SET_TRAINS_DATA_STATUS",
SET_REGION: "SET_REGION", SET_REGION: "SET_REGION",
UPDATE_STATIONS: "UPDATE_STATIONS",
UPDATE_TRAINS: "UPDATE_TRAINS",
UPDATE_TIMETABLES: "UPDATE_TIMETABLES"
} }
export const GETTERS = { export const GETTERS = {
@@ -24,11 +20,9 @@ export const GETTERS = {
trainList: "trainList", trainList: "trainList",
allData: "allData", allData: "allData",
timetableDataStatus: "timetableDataStatus", dispatcherDataSWDRStatus: "dispatcherDataSWDRStatus",
sceneryDataStatus: "sceneryDataStatus", swdrDataStatus: "swdrDataStatus",
dispatcherDataStatus: "dispatcherDataStatus",
trainsDataStatus: "trainsDataStatus", trainsDataStatus: "trainsDataStatus",
dataStatus: "dataStatus",
currentRegion: "currentRegion" currentRegion: "currentRegion"
} }
+9
View File
@@ -26,6 +26,15 @@ export default defineComponent({
hour: "2-digit", hour: "2-digit",
minute: "2-digit" minute: "2-digit"
}) })
},
timestampToString(timestamp: number | null) {
return timestamp
? new Date(timestamp).toLocaleTimeString("pl-PL", {
hour: "2-digit",
minute: "2-digit"
})
: "";
} }
} }
}) })
+3 -2
View File
@@ -30,8 +30,9 @@ const routes: Array<RouteRecordRaw> = [
] ]
const router = createRouter({ const router = createRouter({
scrollBehavior(to, from) { scrollBehavior(to, from) {
if (to.name == "SceneryView")
if (to.name == "SceneryView" && from.name)
return { el: `.app_main` }; return { el: `.app_main` };
if (from.name == "SceneryView" && to.name == "StationsView") if (from.name == "SceneryView" && to.name == "StationsView")
+1 -1
View File
@@ -46,7 +46,7 @@ export default interface Station {
dispatcherIsSupporter: boolean; dispatcherIsSupporter: boolean;
statusTimestamp: number; statusTimestamp: number;
statusTimeString: string; // statusTimeString: string;
statusID: string; statusID: string;
stationTrains?: { stationTrains?: {
+1 -6
View File
@@ -6,13 +6,8 @@ export interface StoreData {
stationList: Station[]; stationList: Station[];
trainList: Train[]; trainList: Train[];
activeTrainCount: number;
activeStationCount: number;
dataConnectionStatus: DataStatus;
timetableDataStatus: DataStatus;
sceneryDataStatus: DataStatus; sceneryDataStatus: DataStatus;
dispatcherDataStatus: DataStatus; dispatcherDataStatus: DataStatus;
trainsDataStatus: DataStatus; trainsDataStatus: DataStatus;
} }
+1 -3
View File
@@ -27,8 +27,6 @@ export default interface Train {
TWR: boolean; TWR: boolean;
SKR: boolean; SKR: boolean;
routeDistance: number; routeDistance: number;
sceneries: string[];
}; };
stopStatus: string;
stopLabel: string;
} }
+6 -6
View File
@@ -1,4 +1,4 @@
export default interface TrainStop { export default interface TrainStop {
stopName: string; stopName: string;
stopNameRAW: string; stopNameRAW: string;
stopType: string; stopType: string;
@@ -6,19 +6,19 @@ export default interface TrainStop {
mainStop: boolean; mainStop: boolean;
arrivalLine: string | null; arrivalLine: string | null;
arrivalTimeString: string | null; // arrivalTimeString: string | null;
arrivalTimestamp: number; arrivalTimestamp: number;
arrivalRealTimeString: string | null; // arrivalRealTimeString: string | null;
arrivalRealTimestamp: number; arrivalRealTimestamp: number;
arrivalDelay: number; arrivalDelay: number;
departureLine: string | null; departureLine: string | null;
departureTimeString: string | null; // departureTimeString: string | null;
departureTimestamp: number; departureTimestamp: number;
departureRealTimeString: string | null; // departureRealTimeString: string | null;
departureRealTimestamp: number; departureRealTimestamp: number;
departureDelay: number; departureDelay: number;
pointId: string; pointId: number;
comments?: any; comments?: any;
+56 -13
View File
@@ -1,19 +1,62 @@
import StationAPIData from "./StationAPIData";
export default interface TrainAPIData { export default interface TrainAPIData {
trainNo: number; trainNo: number;
driverId: number;
mass: number;
length: number;
speed: number;
stockString: string;
signal: string;
distance: number;
connectedTrack: string;
driverName: string; driverName: string;
driverId: number;
driverIsSupporter: boolean; driverIsSupporter: boolean;
station: StationAPIData;
dataSignal: string; currentStationName: string;
dataSceneryConnection: string; currentStationHash: string;
dataDistance: number;
dataCon: string; online: boolean;
dataSpeed: number;
dataMass: number;
dataLength: number;
region: string;
isOnline: boolean;
lastSeen: number; lastSeen: number;
region: string;
timetable?: {
timetableId: number;
category: string;
route: string;
stopList: {
stopName: string;
stopNameRAW: string;
stopType: string;
stopDistance: number;
pointId: number;
mainStop: boolean;
arrivalLine: string;
arrivalTimestamp: number;
arrivalRealTimestamp: number;
arrivalDelay: number;
departureLine: string;
departureTimestamp: number;
departureRealTimestamp: number;
departureDelay: number;
comments?: any;
beginsHere: boolean;
terminatesHere: boolean;
confirmed: boolean;
stopped: boolean;
stopTime: number;
}[];
TWR: boolean;
SKR: boolean;
sceneries: string[];
};
} }
+3 -2
View File
@@ -2,8 +2,9 @@ export const URLs = {
sceneryData: "https://spythere.github.io/api/stationData.json", sceneryData: "https://spythere.github.io/api/stationData.json",
sceneryDataDev: "http://127.0.0.1:8000/data", sceneryDataDev: "http://127.0.0.1:8000/data",
stations: "https://api.td2.info.pl:9640/?method=getStationsOnline", stations: "https://api.td2.info.pl:9640/?method=getStationsOnline",
trains: "https://api.td2.info.pl:9640/?method=getTrainsOnline",
dispatchers: "https://api.td2.info.pl:9640/?method=readFromSWDR&value=getDispatcherStatusList%3B1", dispatchers: "https://api.td2.info.pl:9640/?method=readFromSWDR&value=getDispatcherStatusList%3B1",
stacjownikAPI: "https://stacjownik.herokuapp.com", stacjownikAPI: "https://stacjownik.herokuapp.com",
getTimetableURL: (trainNo: string | number, region = "eu") => `https://api.td2.info.pl:9640/?method=readFromSWDR&value=getTimetable%3B${trainNo}%3B${region}` stacjownikAPIDev: "http://localhost:3001"
// trains: "https://api.td2.info.pl:9640/?method=getTrainsOnline",
// getTimetableURL: (trainNo: string | number, region = "eu") => `https://api.td2.info.pl:9640/?method=readFromSWDR&value=getTimetable%3B${trainNo}%3B${region}`
}; };
+30 -17
View File
@@ -1,5 +1,4 @@
import Station from "../interfaces/Station"; import Train from "../interfaces/Train";
import Timetable from "../interfaces/Timetable";
import TrainStop from "../interfaces/TrainStop"; import TrainStop from "../interfaces/TrainStop";
export const getLocoURL = (locoType: string): string => (`https://rj.td2.info.pl/dist/img/thumbnails/${locoType.includes("EN") ? locoType + "rb" : locoType}.png`) export const getLocoURL = (locoType: string): string => (`https://rj.td2.info.pl/dist/img/thumbnails/${locoType.includes("EN") ? locoType + "rb" : locoType}.png`)
@@ -72,15 +71,7 @@ export const parseSpawns = (spawnString: string) => {
export const getTimestamp = (date: string | null): number => (date ? new Date(date).getTime() : 0); export const getTimestamp = (date: string | null): number => (date ? new Date(date).getTime() : 0);
export const timestampToString = (timestamp: number | null): string => export const getTrainStopStatus = (stopInfo: TrainStop, currentStationName: string, stationName: string) => {
timestamp
? new Date(timestamp).toLocaleTimeString("pl-PL", {
hour: "2-digit",
minute: "2-digit"
})
: "";
export const getTrainStopStatus = (stopInfo: TrainStop, currentStationName: string, station: Station) => {
let stopStatus = "", let stopStatus = "",
stopLabel = "", stopLabel = "",
stopStatusID = -1; stopStatusID = -1;
@@ -89,27 +80,49 @@ export const getTrainStopStatus = (stopInfo: TrainStop, currentStationName: stri
stopStatus = "terminated"; stopStatus = "terminated";
stopLabel = "Skończył bieg"; stopLabel = "Skończył bieg";
stopStatusID = 5; stopStatusID = 5;
} else if (!stopInfo.terminatesHere && stopInfo.confirmed && currentStationName == station.name) { } else if (!stopInfo.terminatesHere && stopInfo.confirmed && currentStationName == stationName) {
stopStatus = "departed"; stopStatus = "departed";
stopLabel = "Odprawiony"; stopLabel = "Odprawiony";
stopStatusID = 2; stopStatusID = 2;
} else if (!stopInfo.terminatesHere && stopInfo.confirmed && currentStationName != station.name) { } else if (!stopInfo.terminatesHere && stopInfo.confirmed && currentStationName != stationName) {
stopStatus = "departed-away"; stopStatus = "departed-away";
stopLabel = "Odjechał"; stopLabel = "Odjechał";
stopStatusID = 4; stopStatusID = 4;
} else if (currentStationName == station.name && !stopInfo.stopped) { } else if (currentStationName == stationName && !stopInfo.stopped) {
stopStatus = "online"; stopStatus = "online";
stopLabel = "Na stacji"; stopLabel = "Na stacji";
stopStatusID = 0; stopStatusID = 0;
} else if (currentStationName == station.name && stopInfo.stopped) { } else if (currentStationName == stationName && stopInfo.stopped) {
stopStatus = "stopped"; stopStatus = "stopped";
stopLabel = "Postój"; stopLabel = "Postój";
stopStatusID = 1; stopStatusID = 1;
} else if (currentStationName != station.name) { } else if (currentStationName != stationName) {
stopStatus = "arriving"; stopStatus = "arriving";
stopLabel = "W drodze"; stopLabel = "W drodze";
stopStatusID = 3; stopStatusID = 3;
} }
return { stopStatus, stopLabel, stopStatusID }; return { stopStatus, stopLabel, stopStatusID };
}; };
export function getScheduledTrain(train: Train, trainStop: TrainStop, stationName: string) {
const timetable = train.timetableData!;
const trainStopStatus = getTrainStopStatus(trainStop, train.currentStationName, stationName);
return {
trainNo: train.trainNo,
driverName: train.driverName,
driverId: train.driverId,
currentStationName: train.currentStationName,
currentStationHash: train.currentStationHash,
category: timetable.category,
beginsAt: timetable.followingStops[0].stopNameRAW,
terminatesAt: timetable.followingStops[timetable.followingStops.length - 1].stopNameRAW,
nearestStop: "",
stopInfo: trainStop,
stopLabel: trainStopStatus.stopLabel,
stopStatus: trainStopStatus.stopStatus,
stopStatusID: trainStopStatus.stopStatusID
}
}
+173 -394
View File
@@ -7,19 +7,16 @@ import axios from "axios";
import Station from "@/scripts/interfaces/Station"; import Station from "@/scripts/interfaces/Station";
import Train from "@/scripts/interfaces/Train"; import Train from "@/scripts/interfaces/Train";
import TrainStop from "@/scripts/interfaces/TrainStop";
import { StoreData } from "@/scripts/interfaces/StoreData"; import { StoreData } from "@/scripts/interfaces/StoreData";
import TimetableAPIData from '@/scripts/interfaces/api/TimetableAPIData';
import StationAPIData from '@/scripts/interfaces/api/StationAPIData'; import StationAPIData from '@/scripts/interfaces/api/StationAPIData';
import TrainAPIData from '@/scripts/interfaces/api/TrainAPIData'; import TrainAPIData from '@/scripts/interfaces/api/TrainAPIData';
import Timetable from '@/scripts/interfaces/Timetable';
import { ACTIONS, MUTATIONS } from "@/constants/storeConstants"; import { ACTIONS, MUTATIONS } from "@/constants/storeConstants";
import { DataStatus } from "@/scripts/enums/DataStatus"; import { DataStatus } from "@/scripts/enums/DataStatus";
import { getLocoURL, getStatusID, getStatusTimestamp, getTimestamp, getTrainStopStatus, parseSpawns, timestampToString } from "@/scripts/utils/storeUtils"; import { getLocoURL, getScheduledTrain, getStatusID, getStatusTimestamp, parseSpawns } from "@/scripts/utils/storeUtils";
import { URLs } from '@/scripts/utils/apiURLs'; import { URLs } from '@/scripts/utils/apiURLs';
import ScheduledTrain from '@/scripts/interfaces/ScheduledTrain'; import ScheduledTrain from '@/scripts/interfaces/ScheduledTrain';
import StationRoutes from '@/scripts/interfaces/StationRoutes'; import StationRoutes from '@/scripts/interfaces/StationRoutes';
@@ -27,7 +24,7 @@ import StationRoutes from '@/scripts/interfaces/StationRoutes';
export interface State { export interface State {
stationList: Station[], stationList: Station[],
trainList: Train[], trainList: Train[],
timetableList: Timetable[], // timetableList: Timetable[],
sceneryData: any[][], sceneryData: any[][],
@@ -68,13 +65,14 @@ interface StationJSONData {
unavailable: boolean; unavailable: boolean;
} }
export const key: InjectionKey<Store<State>> = Symbol() export const key: InjectionKey<Store<State>> = Symbol()
export const store = createStore<State>({ export const store = createStore<State>({
state: () => ({ state: () => ({
stationList: [], stationList: [],
trainList: [], trainList: [],
timetableList: [], // timetableList: [],
sceneryData: [], sceneryData: [],
@@ -100,247 +98,213 @@ export const store = createStore<State>({
stationList: state.stationList, stationList: state.stationList,
trainList: state.trainList, trainList: state.trainList,
activeTrainCount: state.trainCount,
activeStationCount: state.stationCount,
dataConnectionStatus: state.dataConnectionStatus,
timetableDataStatus: state.timetableDataStatus,
sceneryDataStatus: state.sceneryDataStatus, sceneryDataStatus: state.sceneryDataStatus,
dispatcherDataStatus: state.dispatcherDataStatus, dispatcherDataStatus: state.dispatcherDataStatus,
trainsDataStatus: state.trainsDataStatus trainsDataStatus: state.trainsDataStatus
}), }),
timetableDataStatus: (state): DataStatus => state.timetableDataStatus,
sceneryDataStatus: (state): DataStatus => state.sceneryDataStatus, sceneryDataStatus: (state): DataStatus => state.sceneryDataStatus,
trainsDataStatus: (state): DataStatus => state.trainsDataStatus, trainsDataStatus: (state): DataStatus => state.trainsDataStatus,
dataStatus: (state): DataStatus => state.dataConnectionStatus, dataStatus: (state): DataStatus => state.dataConnectionStatus,
currentRegion: (state): { id: string; value: string } => state.region currentRegion: (state): { id: string; value: string } => state.region
}, },
actions: { actions: {
async synchronizeData({ commit, dispatch, state }) { async synchronizeData({ commit, dispatch, state }) {
if (state.listenerLaunched) return; if (state.listenerLaunched) return;
// Nowy parametr żądania co godzinę await dispatch(ACTIONS.loadStaticStationData);
const sceneryData: StationJSONData = await (await axios.get(`${URLs.sceneryData}?time=${Math.floor(Date.now() / 1800000)}`)).data; await dispatch(ACTIONS.fetchOnlineData);
commit(MUTATIONS.SET_TIMETABLE_DATA_STATUS, DataStatus.Loading);
commit(MUTATIONS.SET_SCENERY_DATA, sceneryData);
commit(MUTATIONS.SET_SCENERY_DATA_STATUS, DataStatus.Loaded);
dispatch(ACTIONS.fetchOnlineData);
setInterval(() => dispatch(ACTIONS.fetchOnlineData), Math.floor(Math.random() * 5000) + 25000); setInterval(() => dispatch(ACTIONS.fetchOnlineData), Math.floor(Math.random() * 5000) + 25000);
}, },
async fetchOnlineData({ commit, dispatch }) { async loadStaticStationData({ commit }) {
commit(MUTATIONS.SET_DATA_CONNECTION_STATUS, DataStatus.Loading); // Nowy parametr żądania co godzinę
const sceneryData: StationJSONData = await (await axios.get(`${URLs.sceneryData}?time=${Math.floor(Date.now() / 1800000)}`)).data;
Promise.all([axios.get(URLs.stations), axios.get(URLs.trains), axios.get(URLs.dispatchers)]) if (!sceneryData)
.then(async response => { commit(MUTATIONS.SET_SCENERY_DATA_STATUS, DataStatus.Error);
const onlineStationsData: { success: boolean, message: StationAPIData[] } = response[0].data; else
const onlineTrainsData: { success: boolean, message: TrainAPIData[] } = await response[1].data; commit(MUTATIONS.SET_SCENERY_DATA, sceneryData);
const onlineDispatchersData: { success: boolean, message: string[][] } = await response[2].data;
if (!onlineStationsData.success) {
// commit(MUTATIONS.SET_DATA_CONNECTION_STATUS, DataStatus.Error);
commit(MUTATIONS.SET_SCENERY_DATA_STATUS, DataStatus.Error);
return;
}
commit(MUTATIONS.SET_SCENERY_DATA_STATUS, DataStatus.Loaded);
commit(MUTATIONS.SET_DISPATCHER_DATA_STATUS, onlineDispatchersData.success ? DataStatus.Loaded : DataStatus.Warning);
commit(MUTATIONS.SET_TRAINS_DATA_STATUS, onlineTrainsData.success ? DataStatus.Loaded : DataStatus.Warning);
const updatedStationList: Station['onlineInfo'][] = onlineStationsData.message.reduce((acc, station) => {
if (station.region !== this.state.region.id || !station.isOnline) return acc;
const stationStatus = onlineDispatchersData.success ? onlineDispatchersData.message.find((status: string[]) => status[0] == station.stationHash && status[1] == this.state.region.id) : -1;
const statusTimestamp = getStatusTimestamp(stationStatus);
const statusID = getStatusID(stationStatus);
const stationTrains = onlineTrainsData.success ? onlineTrainsData.message
.filter(train => train.region === this.state.region.id && train.isOnline && train.station.stationName === station.stationName)
.map(train => ({ driverName: train.driverName, trainNo: train.trainNo })) : [];
acc.push({
name: station.stationName,
hash: station.stationHash,
maxUsers: station.maxUsers,
currentUsers: station.currentUsers,
spawns: parseSpawns(station.spawnString),
dispatcherName: station.dispatcherName,
dispatcherRate: station.dispatcherRate,
dispatcherId: station.dispatcherId,
dispatcherExp: station.dispatcherExp,
dispatcherIsSupporter: station.dispatcherIsSupporter,
stationTrains,
statusTimestamp,
statusID,
statusTimeString: timestampToString(statusTimestamp),
});
return acc;
}, [] as Station['onlineInfo'][]);
const updatedTrainList = onlineTrainsData.success ? await Promise.all(
onlineTrainsData.message
.filter(train => train.region === this.state.region.id)
.map(async train => {
const locoType = train.dataCon.split(";") ? train.dataCon.split(";")[0] : train.dataCon;
return {
trainNo: train.trainNo,
mass: train.dataMass,
length: train.dataLength,
speed: train.dataSpeed,
distance: train.dataDistance,
signal: train.dataSignal,
online: train.isOnline,
driverId: train.driverId,
driverName: train.driverName,
currentStationName: train.station.stationName,
currentStationHash: train.station.stationHash,
connectedTrack: train.dataSceneryConnection,
locoType,
locoURL: getLocoURL(locoType),
cars: train.dataCon.split(";").filter((train, i) => i > 0) || [],
};
})
) : [];
// Pass reduced lists to mutations
commit(MUTATIONS.UPDATE_STATIONS, updatedStationList);
commit(MUTATIONS.UPDATE_TRAINS, updatedTrainList);
// Statuses
commit(MUTATIONS.SET_DATA_CONNECTION_STATUS, DataStatus.Loaded);
// commit(MUTATIONS.SET_TIMETABLE_DATA_STATUS, DataStatus.Loading);
dispatch(ACTIONS.fetchTimetableData);
})
.catch(() => {
commit(MUTATIONS.SET_DATA_CONNECTION_STATUS, DataStatus.Error);
});
}, },
async fetchTimetableData({ commit }) { async fetchOnlineData({ commit }) {
let warnings = 0; // Pobierz dane o pociągach i rozkładach jazdy z API Stacjownika
const trainsAPIData: { response: TrainAPIData[], errorMessage?: string } = (await axios.get(`${URLs.stacjownikAPI}/api/getActiveTrainList`)).data;
const timetableList = this.state.trainList.reduce(async (acc: Promise<Timetable[]>, train: Train) => { // Pobierz dane o sceneriach online i o dyżurnych
const data: { success: boolean; message: TimetableAPIData } = await (await axios.get(URLs.getTimetableURL(train.trainNo, this.state.region.id))).data; const dispatchersAPIData: { success: boolean, message: string[][] } = await (await axios.get(URLs.dispatchers)).data;
const stationsAPIData: { success: boolean, message: StationAPIData[] } = await (await axios.get(URLs.stations)).data;
if (!data.success) { if (!stationsAPIData || !stationsAPIData.success) {
warnings++; commit(MUTATIONS.SET_SCENERY_DATA_STATUS, DataStatus.Error);
return;
}
(await acc).push({ commit(MUTATIONS.SET_SCENERY_DATA_STATUS, DataStatus.Loaded);
trainNo: train.trainNo,
success: false if (!dispatchersAPIData || !dispatchersAPIData.success)
commit(MUTATIONS.SET_DISPATCHER_DATA_STATUS, DataStatus.Warning)
if (!trainsAPIData || !trainsAPIData.response)
commit(MUTATIONS.SET_TRAINS_DATA_STATUS, DataStatus.Warning);
// Zaktualizuj listę pociągów
const updatedTrainList: Train[] =
trainsAPIData?.response
.filter(train => train.region === this.state.region.id && train.online)
.map(train => {
const stock = train.stockString.split(";");
const locoType = stock ? stock[0] : train.stockString;
const timetable = train.timetable;
return {
trainNo: train.trainNo,
mass: train.mass,
length: train.length,
speed: train.speed,
region: train.region,
distance: train.distance,
signal: train.signal,
online: train.online,
driverId: train.driverId,
driverName: train.driverName,
currentStationName: train.currentStationName,
currentStationHash: train.currentStationHash,
connectedTrack: train.connectedTrack,
locoType,
locoURL: getLocoURL(locoType),
cars: stock.slice(1),
timetableData: timetable ? {
timetableId: timetable.timetableId,
SKR: timetable.SKR,
TWR: timetable.TWR,
route: timetable.route,
category: timetable.category,
followingStops: timetable.stopList,
routeDistance: timetable.stopList[timetable.stopList.length - 1].stopDistance,
sceneries: timetable.sceneries
} : undefined
};
}) || []
const onlineStationNames: string[] = [];
stationsAPIData.message.forEach((stationAPI) => {
if (stationAPI.region !== this.state.region.id || !stationAPI.isOnline) return;
onlineStationNames.push(stationAPI.stationName);
const stationName = stationAPI.stationName.toLowerCase();
const station = this.state.stationList.find(s => s.name == stationAPI.stationName);
const stationStatus = dispatchersAPIData.success ? dispatchersAPIData.message.find((status: string[]) => status[0] == stationAPI.stationHash && status[1] == this.state.region.id) : -1;
const statusTimestamp = getStatusTimestamp(stationStatus);
const statusID = getStatusID(stationStatus);
const stationTrains = trainsAPIData.response
.filter(train => train.region === this.state.region.id && train.online && train.currentStationName === stationAPI.stationName)
.map(train => ({ driverName: train.driverName, trainNo: train.trainNo }));
station?.generalInfo?.checkpoints.forEach(cp => cp.scheduledTrains.length = 0);
const scheduledTrains: ScheduledTrain[] = updatedTrainList.reduce((acc: ScheduledTrain[], train) => {
if (!train.timetableData) return acc;
const timetable = train.timetableData;
if (!timetable.sceneries.includes(stationAPI.stationHash)) return acc;
const stopInfoIndex = timetable.followingStops.findIndex(stop => {
const stopName = stop.stopNameRAW.toLowerCase();
// if (stop.stopName == "ARKADIA ZDRÓJ" && station.name == "Arkadia Zdrój 2019" && stop.pointId != "1583014379097") return false;
// if (stop.stopName == "ARKADIA ZDRÓJ" && station.name == "Arkadia Zdrój 2012" && stop.pointId != "1519258642187") return false;
if (stationName === stopName) return true;
if (stopName.includes(stationName) && !stop.stopName.includes("po.") && !stop.stopName.includes("podg.")) return true;
if (stationName.includes(stopName) && !stop.stopName.includes("po.") && !stop.stopName.includes("podg.")) return true;
if (stopName.includes("podg.") && stopName.split(", podg.")[0] && stationName.includes(stopName.split(", podg.")[0])) return true;
if (station?.generalInfo
&& station.generalInfo.checkpoints
&& station.generalInfo.checkpoints.length > 0
&& station.generalInfo.checkpoints.some(cp => cp.checkpointName.toLowerCase().includes(stop.stopNameRAW.toLowerCase())))
return true;
return false;
}); });
return acc; if (stopInfoIndex == -1) return acc;
}
const timetable = data.message; const trainStop = timetable.followingStops[stopInfoIndex];
const trainInfo = timetable.trainInfo; const scheduledStopTrain = getScheduledTrain(train, trainStop, stationAPI.stationName);
if (!timetable || !trainInfo) return acc; if (station && station.generalInfo?.checkpoints && station.generalInfo.checkpoints.length > 0) {
for (const checkpoint of station.generalInfo.checkpoints) {
timetable.followingStops
.filter(trainStop => trainStop.stopNameRAW.toLowerCase() === checkpoint.checkpointName.toLowerCase())
.forEach(trainCheckpointStop => {
const scheduledCheckpointTrain = getScheduledTrain(train, trainCheckpointStop, stationAPI.stationName);
let lastArrivalLine = ""; checkpoint.scheduledTrains.push(scheduledCheckpointTrain);
});
const followingStops: TrainStop[] = timetable.stopPoints.reduce((stopsAcc: TrainStop[], point) => {
if (point.pointNameRAW.toLowerCase().includes("sbl")) {
if (point.arrivalLine && !point.arrivalLine.toLocaleLowerCase().includes("sbl"))
lastArrivalLine = point.arrivalLine;
if (point.departureLine && !point.departureLine.toLocaleLowerCase().includes("sbl")) {
stopsAcc[stopsAcc.length - 1].departureLine = point.departureLine
} }
return stopsAcc;
} }
const arrivalTimestamp = getTimestamp(point.arrivalTime); acc.push(scheduledStopTrain);
const arrivalRealTimestamp = getTimestamp(point.arrivalRealTime); return acc;
const departureTimestamp = getTimestamp(point.departureTime);
const departureRealTimestamp = getTimestamp(point.departureRealTime);
let arrivalLine = lastArrivalLine || point.arrivalLine;
if (lastArrivalLine != "")
lastArrivalLine = "";
stopsAcc.push({
stopName: point.pointName,
stopNameRAW: point.pointNameRAW,
stopType: point.pointStopType,
stopDistance: point.pointDistance,
pointId: point.pointId,
comments: point.comments,
mainStop: point.pointName.includes("strong"),
arrivalLine,
arrivalTimeString: timestampToString(arrivalTimestamp),
arrivalTimestamp: arrivalTimestamp,
arrivalRealTimeString: timestampToString(arrivalRealTimestamp),
arrivalRealTimestamp: arrivalRealTimestamp,
arrivalDelay: point.arrivalDelay,
departureLine: point.departureLine,
departureTimeString: timestampToString(departureTimestamp),
departureTimestamp: departureTimestamp,
departureRealTimeString: timestampToString(departureRealTimestamp),
departureRealTimestamp: departureRealTimestamp,
departureDelay: point.departureDelay,
beginsHere: arrivalTimestamp == 0,
terminatesHere: departureTimestamp == 0,
confirmed: point.confirmed,
stopped: point.isStopped,
stopTime: point.pointStopTime
});
return stopsAcc;
}, []); }, []);
(await acc).push({ const onlineInfo = {
data: { name: stationAPI.stationName,
trainNo: train.trainNo, hash: stationAPI.stationHash,
driverName: train.driverName, maxUsers: stationAPI.maxUsers,
driverId: train.driverId, currentUsers: stationAPI.currentUsers,
currentStationName: train.currentStationName, spawns: parseSpawns(stationAPI.spawnString),
currentStationHash: train.currentStationHash, dispatcherName: stationAPI.dispatcherName,
timetableId: trainInfo.timetableId, dispatcherRate: stationAPI.dispatcherRate,
category: trainInfo.trainCategoryCode, dispatcherId: stationAPI.dispatcherId,
route: trainInfo.route, dispatcherExp: stationAPI.dispatcherExp,
TWR: trainInfo.twr, dispatcherIsSupporter: stationAPI.dispatcherIsSupporter,
SKR: trainInfo.skr, stationTrains,
routeDistance: timetable.stopPoints[timetable.stopPoints.length - 1].pointDistance, statusTimestamp,
followingStops, statusID,
followingSceneries: trainInfo.sceneries scheduledTrains,
}, // statusTimeString: timestampToString(statusTimestamp),
}
trainNo: train.trainNo, if (!station) {
success: true this.state.stationList.push({
name: stationAPI.stationName,
onlineInfo
})
return;
}
station.onlineInfo = { ...onlineInfo };
});
this.state.stationList
.filter(station => !onlineStationNames.includes(station.name) && station.onlineInfo)
.forEach(offlineStation => {
offlineStation.onlineInfo = undefined;
}); });
return acc; this.state.trainList = updatedTrainList;
}, Promise.resolve([])); },
commit(MUTATIONS.UPDATE_TIMETABLES, (await timetableList));
commit(MUTATIONS.SET_TIMETABLE_DATA_STATUS, warnings == 0 ? DataStatus.Loaded : DataStatus.Warning);
}
}, },
mutations: { mutations: {
SET_SCENERY_DATA(state, data: StationJSONData[]) { SET_SCENERY_DATA(state, data: StationJSONData[]) {
state.stationList = data.map(stationData => ({ state.stationList = data.map(stationData => ({
name: stationData.name, name: stationData.name,
@@ -349,7 +313,7 @@ export const store = createStore<State>({
routes: stationData.routes?.split(";").filter(routeString => routeString).reduce((acc, routeString) => { routes: stationData.routes?.split(";").filter(routeString => routeString).reduce((acc, routeString) => {
const specs1 = routeString.split("_")[0]; const specs1 = routeString.split("_")[0];
const isInternal = specs1.startsWith('!'); const isInternal = specs1.startsWith('!');
const name = isInternal ? specs1.replace("!", "") : specs1; const name = isInternal ? specs1.replace("!", "") : specs1;
const specs2 = routeString.split("_")[1].split(""); const specs2 = routeString.split("_")[1].split("");
const twoWay = specs2[0] == "2"; const twoWay = specs2[0] == "2";
@@ -366,7 +330,7 @@ export const store = createStore<State>({
: 'oneWayNoCatenaryRouteNames'; : 'oneWayNoCatenaryRouteNames';
acc[twoWay ? 'twoWay' : 'oneWay'].push({ name, SBL, TWB, catenary, isInternal, tracks: twoWay ? 2 : 1 }); acc[twoWay ? 'twoWay' : 'oneWay'].push({ name, SBL, TWB, catenary, isInternal, tracks: twoWay ? 2 : 1 });
if(!isInternal) acc[propName].push(name); if (!isInternal) acc[propName].push(name);
if (SBL) acc['sblRouteNames'].push(name); if (SBL) acc['sblRouteNames'].push(name);
@@ -383,21 +347,13 @@ export const store = createStore<State>({
checkpoints: stationData.checkpoints ? stationData.checkpoints.split(";").map(sub => ({ checkpointName: sub, scheduledTrains: [] })) : [], checkpoints: stationData.checkpoints ? stationData.checkpoints.split(";").map(sub => ({ checkpointName: sub, scheduledTrains: [] })) : [],
} }
})); }));
},
SET_DATA_CONNECTION_STATUS(state, status: DataStatus) {
state.dataConnectionStatus = status;
}, },
SET_SCENERY_DATA_STATUS(state, status: DataStatus) { SET_SCENERY_DATA_STATUS(state, status: DataStatus) {
state.sceneryDataStatus = status; state.sceneryDataStatus = status;
}, },
SET_TIMETABLE_DATA_STATUS(state, status: DataStatus) {
state.timetableDataStatus = status;
},
SET_DISPATCHER_DATA_STATUS(state, status: DataStatus) { SET_DISPATCHER_DATA_STATUS(state, status: DataStatus) {
state.dispatcherDataStatus = status; state.dispatcherDataStatus = status;
}, },
@@ -406,187 +362,10 @@ export const store = createStore<State>({
state.trainsDataStatus = status; state.trainsDataStatus = status;
}, },
SET_REGION(state, region: { id: string; value: string }) { SET_REGION(state, region: { id: string; value: string }) {
state.region = region; state.region = region;
}, },
UPDATE_STATIONS(state, updatedStationList: Station['onlineInfo'][]) {
state.stationList = state.stationList.reduce((acc: Station[], station) => {
const onlineStationData = updatedStationList.find(updatedStation => updatedStation?.name === station.name);
const listedStationData = state.stationList.find(data => data.name === station.name);
if (onlineStationData)
acc.push({
name: station.name,
generalInfo: station.generalInfo,
onlineInfo: {
...onlineStationData,
scheduledTrains: station.onlineInfo?.scheduledTrains || []
},
});
else if (listedStationData)
acc.push({
...station,
onlineInfo: undefined
});
return acc;
}, [] as Station[]);
updatedStationList
.filter(uStation => !state.stationList.some(station => uStation?.name === station.name))
.forEach(uStation => {
if (!uStation) return;
state.stationList.push({
name: uStation.name,
onlineInfo: {
...uStation,
scheduledTrains: [],
stationTrains: uStation.stationTrains || []
},
generalInfo: undefined
});
});
state.stationCount = state.stationList.filter(station => station.onlineInfo).length;
state.dataConnectionStatus = DataStatus.Loaded;
},
UPDATE_TRAINS(state, updatedTrainList: any[]) {
state.trainList = updatedTrainList.reduce((acc, updatedTrain) => {
const trainData = state.trainList.find(train => train.trainNo === updatedTrain.trainNo);
if (trainData) acc.push({ ...trainData, ...updatedTrain });
else acc.push({ ...updatedTrain });
return acc;
}, [] as Train[]);
state.trainCount = state.trainList.length;
state.dataConnectionStatus = DataStatus.Loaded;
},
UPDATE_TIMETABLES(state, timetableList: Timetable[]) {
state.stationList = state.stationList.map(station => {
const stationName = station.name.toLowerCase();
const scheduledTrains: ScheduledTrain[] = timetableList.reduce((acc: ScheduledTrain[], timetable: Timetable) => {
if (!timetable.data)
return acc;
if (!timetable.data.followingSceneries.includes(station.onlineInfo?.hash || "")) return acc;
const stopInfoIndex = timetable.data.followingStops.findIndex(stop => {
const stopName = stop.stopNameRAW.toLowerCase();
// if (stop.stopName == "ARKADIA ZDRÓJ" && station.name == "Arkadia Zdrój 2019" && stop.pointId != "1583014379097") return false;
// if (stop.stopName == "ARKADIA ZDRÓJ" && station.name == "Arkadia Zdrój 2012" && stop.pointId != "1519258642187") return false;
if (stationName === stopName) return true;
if (stopName.includes(stationName) && !stop.stopName.includes("po.") && !stop.stopName.includes("podg.")) return true;
if (stationName.includes(stopName) && !stop.stopName.includes("po.") && !stop.stopName.includes("podg.")) return true;
if (stopName.includes("podg.") && stopName.split(", podg.")[0] && stationName.includes(stopName.split(", podg.")[0])) return true;
if (station.generalInfo
&& station.generalInfo.checkpoints
&& station.generalInfo.checkpoints.length > 0
&& station.generalInfo.checkpoints.some(cp => cp.checkpointName.toLowerCase().includes(stop.stopNameRAW.toLowerCase())))
return true;
return false;
});
if (stopInfoIndex == -1) return acc;
const trainStop = timetable.data.followingStops[stopInfoIndex];
const trainStopStatus = getTrainStopStatus(trainStop, timetable.data.currentStationName, station);
acc.push({
trainNo: timetable.data.trainNo,
driverName: timetable.data.driverName,
driverId: timetable.data.driverId,
currentStationName: timetable.data.currentStationName,
currentStationHash: timetable.data.currentStationHash,
category: timetable.data.category,
beginsAt: timetable.data.followingStops[0].stopNameRAW,
terminatesAt: timetable.data.followingStops[timetable.data.followingStops.length - 1].stopNameRAW,
nearestStop: "",
stopInfo: trainStop,
stopLabel: trainStopStatus.stopLabel,
stopStatus: trainStopStatus.stopStatus,
stopStatusID: trainStopStatus.stopStatusID
});
return acc;
}, []);
if (station.generalInfo && station.generalInfo.checkpoints.length > 0) {
station.generalInfo.checkpoints.forEach(cp => (cp.scheduledTrains.length = 0));
for (const checkpoint of station.generalInfo.checkpoints) {
timetableList.forEach(timetable => {
if (!timetable.data) return;
if (!timetable.data.followingSceneries.includes(station.onlineInfo?.hash || "")) return;
const timetableData = timetable.data;
timetableData.followingStops
.filter(trainStop => trainStop.stopNameRAW.toLowerCase() === checkpoint.checkpointName.toLowerCase())
.forEach(trainStop => {
const trainStopStatus = getTrainStopStatus(trainStop, timetableData.currentStationName, station);
checkpoint.scheduledTrains.push({
trainNo: timetable.trainNo,
driverName: timetableData.driverName,
driverId: timetableData.driverId,
currentStationName: timetableData.currentStationName,
currentStationHash: timetableData.currentStationHash,
category: timetableData.category,
beginsAt: timetableData.followingStops[0].stopNameRAW,
terminatesAt: timetableData.followingStops[timetableData.followingStops.length - 1].stopNameRAW,
nearestStop: "",
stopInfo: trainStop,
stopLabel: trainStopStatus.stopLabel,
stopStatus: trainStopStatus.stopStatus,
stopStatusID: trainStopStatus.stopStatusID
});
});
});
}
}
return {
...station,
onlineInfo: station.onlineInfo ? {
...station.onlineInfo,
scheduledTrains
} : undefined
};
});
state.trainList = state.trainList.reduce((acc, train) => {
const timetable = timetableList.find(tt => tt.data && tt.trainNo === train.trainNo && tt.data.driverId === train.driverId);
if (!train.online && !timetable) return acc;;
const trainStopData = state.stationList
.find(station => station.name === train.currentStationName)
?.onlineInfo?.scheduledTrains?.find(stationTrain => stationTrain.trainNo === train.trainNo);
acc.push({ ...train, timetableData: timetable?.data, stopStatus: trainStopData?.stopStatus || "", stopLabel: trainStopData?.stopLabel || "" });
return acc;
}, [] as Train[]);
state.timetableDataStatus = DataStatus.Loaded;
}
} }
}) })
+13 -13
View File
@@ -1,6 +1,6 @@
<template> <template>
<div class="scenery-view"> <div class="scenery-view">
<div class="scenery-offline" v-if="!stationInfo && isDataLoaded && isComponentVisible"> <div class="scenery-offline" v-if="!stationInfo && isComponentVisible">
<div>{{ $t('scenery.no-scenery') }}</div> <div>{{ $t('scenery.no-scenery') }}</div>
<action-button> <action-button>
@@ -77,8 +77,8 @@ export default defineComponent({
viewMode: 'info', viewMode: 'info',
stationInfo: {} as (Station | undefined), stationInfo: {} as Station | undefined,
onlineFrom: -1 onlineFrom: -1,
}), }),
setup() { setup() {
@@ -91,12 +91,11 @@ export default defineComponent({
const isComponentVisible = computed(() => route.path === '/scenery'); const isComponentVisible = computed(() => route.path === '/scenery');
const isDataLoaded = computed(() => data.value.dataConnectionStatus === DataStatus.Loaded); const stationInfo = computed(() => {
return data.value.stationList.find(
const stationInfo = computed(() => { (station) => station.name === route.query.station?.toString().replace(/_/g, ' ')
return data.value.stationList.find((station) => station.name === route.query.station?.toString().replace(/_/g, ' ')) );
}) });
// const onlineFrom = computed(async () => { // const onlineFrom = computed(async () => {
// return await (await axios.get(`${URLs.stacjownikAPI}?name=${route.query.station}&historyCount=0`)).data; // return await (await axios.get(`${URLs.stacjownikAPI}?name=${route.query.station}&historyCount=0`)).data;
@@ -107,8 +106,7 @@ export default defineComponent({
currentRegion: computed(() => store.getters[GETTERS.currentRegion]), currentRegion: computed(() => store.getters[GETTERS.currentRegion]),
timetableOnly, timetableOnly,
isComponentVisible, isComponentVisible,
isDataLoaded, stationInfo,
stationInfo
}; };
}, },
@@ -126,10 +124,12 @@ export default defineComponent({
// this.stationInfo = (this.$store.getters[GETTERS.allData] as StoreData).stationList.find((station) => station.name === this.$route.query.station?.toString().replace(/_/g, ' ')) // this.stationInfo = (this.$store.getters[GETTERS.allData] as StoreData).stationList.find((station) => station.name === this.$route.query.station?.toString().replace(/_/g, ' '))
}, },
async activated() { async activated() {
if (this.currentRegion.id != 'eu' && this.viewMode == 'history') this.viewMode = 'info'; if (this.currentRegion.id != 'eu' && this.viewMode == 'history') this.viewMode = 'info';
const onlineFrom = await (await axios.get(`${URLs.stacjownikAPI}/api/getSceneryHistory?name=${this.$route.query.station}&historyCount=0`)).data; const onlineFrom = await (
await axios.get(`${URLs.stacjownikAPI}/api/getSceneryHistory?name=${this.$route.query.station}&historyCount=0`)
).data;
}, },
}); });
</script> </script>
-14
View File
@@ -90,18 +90,6 @@ export default defineComponent({
return filterManager.getFilteredStationList(store.getters[GETTERS.stationList]); return filterManager.getFilteredStationList(store.getters[GETTERS.stationList]);
}); });
const getStatusClass = computed(() => {
if (data.value.dataConnectionStatus == DataStatus.Loading) return 'loading';
if (data.value.dataConnectionStatus == DataStatus.Error) return 'error';
return 'success';
});
const timetableDataStatusClass = computed(() => {
if (data.value.timetableDataStatus == DataStatus.Loading) return 'loading';
if (data.value.timetableDataStatus == DataStatus.Error) return 'error';
return 'success';
});
const focusedStationInfo = computed(() => const focusedStationInfo = computed(() =>
computedStations.value.find((station) => station.name === focusedStationName) computedStations.value.find((station) => station.name === focusedStationName)
); );
@@ -110,8 +98,6 @@ export default defineComponent({
data, data,
computedStations, computedStations,
filterManager, filterManager,
getStatusClass,
timetableDataStatusClass,
focusedStationName, focusedStationName,
focusedStationInfo, focusedStationInfo,
}; };