Compare commits

..

7 Commits

Author SHA1 Message Date
Spythere 5c4c486643 Wersja 1.21.0
Wersja 1.21.0
2024-02-12 15:14:50 +01:00
Spythere 755c729a9b brakujące tłumaczenia; poprawki 2024-02-12 14:58:59 +01:00
Spythere 3ac8d60c5c filtry aktywnych RJ 2024-02-11 15:30:19 +01:00
Spythere dcff3b088f poprawki filtrowania statusów 2024-02-10 23:11:13 +01:00
Spythere 90b2099955 checkpointy; hotfixy 2024-02-10 22:42:35 +01:00
Spythere fc0c04ec9d bump: 1.21.0 2024-02-10 01:31:08 +01:00
Spythere 41b335555a wyświetlanie RJ dla scenerii offline 2024-02-10 01:30:43 +01:00
27 changed files with 248 additions and 193 deletions
+1 -1
View File
@@ -1,6 +1,6 @@
{ {
"name": "stacjownik", "name": "stacjownik",
"version": "1.20.5", "version": "1.21.0",
"private": true, "private": true,
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",
+2 -2
View File
@@ -100,8 +100,8 @@ export default defineComponent({
}, },
onlineDispatchersCount() { onlineDispatchersCount() {
return this.store.onlineSceneryList.filter( return this.store.activeSceneryList.filter(
(scenery) => scenery.region == this.store.region.id (scenery) => scenery.region == this.store.region.id && scenery.dispatcherId != -1
).length; ).length;
}, },
+1 -1
View File
@@ -85,7 +85,7 @@ export default defineComponent({
regionList() { regionList() {
return regionsJSON.map((region) => { return regionsJSON.map((region) => {
const regionStationCount = this.store.onlineSceneryList.filter( const regionStationCount = this.store.activeSceneryList.filter(
(scenery) => scenery.region == region.id (scenery) => scenery.region == region.id
).length; ).length;
@@ -61,6 +61,9 @@ export default defineComponent({
case Status.ActiveDispatcher.UNKNOWN: case Status.ActiveDispatcher.UNKNOWN:
return 'unknown'; return 'unknown';
case Status.ActiveDispatcher.FREE:
return 'free';
default: default:
if (this.dispatcherTimestamp != null && this.dispatcherStatus >= Date.now() + 25500000) if (this.dispatcherTimestamp != null && this.dispatcherStatus >= Date.now() + 25500000)
return 'no-limit'; return 'no-limit';
@@ -76,7 +76,7 @@ import Station from '../../scripts/interfaces/Station';
import Loading from '../Global/Loading.vue'; import Loading from '../Global/Loading.vue';
import styleMixin from '../../mixins/styleMixin'; import styleMixin from '../../mixins/styleMixin';
import listObserverMixin from '../../mixins/listObserverMixin'; import listObserverMixin from '../../mixins/listObserverMixin';
import { OnlineScenery } from '../../store/typings'; import { ActiveScenery } from '../../store/typings';
import { API } from '../../typings/api'; import { API } from '../../typings/api';
import { Status } from '../../typings/common'; import { Status } from '../../typings/common';
import http from '../../http'; import http from '../../http';
@@ -90,7 +90,7 @@ export default defineComponent({
type: Object as PropType<Station> type: Object as PropType<Station>
}, },
onlineScenery: { onlineScenery: {
type: Object as PropType<OnlineScenery> type: Object as PropType<ActiveScenery>
} }
}, },
+2 -2
View File
@@ -15,7 +15,7 @@
<script lang="ts"> <script lang="ts">
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import Station from '../../scripts/interfaces/Station'; import Station from '../../scripts/interfaces/Station';
import { OnlineScenery } from '../../store/typings'; import { ActiveScenery } from '../../store/typings';
export default defineComponent({ export default defineComponent({
props: { props: {
@@ -29,7 +29,7 @@ export default defineComponent({
}, },
onlineScenery: { onlineScenery: {
type: Object as PropType<OnlineScenery> type: Object as PropType<ActiveScenery>
} }
} }
}); });
+2 -2
View File
@@ -90,7 +90,7 @@ import SceneryInfoUserList from './SceneryInfo/SceneryInfoUserList.vue';
import SceneryInfoSpawnList from './SceneryInfo/SceneryInfoSpawnList.vue'; import SceneryInfoSpawnList from './SceneryInfo/SceneryInfoSpawnList.vue';
import SceneryInfoRoutes from './SceneryInfo/SceneryInfoRoutes.vue'; import SceneryInfoRoutes from './SceneryInfo/SceneryInfoRoutes.vue';
import Station from '../../scripts/interfaces/Station'; import Station from '../../scripts/interfaces/Station';
import { OnlineScenery } from '../../store/typings'; import { ActiveScenery } from '../../store/typings';
export default defineComponent({ export default defineComponent({
components: { components: {
@@ -106,7 +106,7 @@ export default defineComponent({
}, },
onlineScenery: { onlineScenery: {
type: Object as PropType<OnlineScenery> type: Object as PropType<ActiveScenery>
} }
} }
}); });
@@ -1,6 +1,6 @@
<template> <template>
<section class="info-dispatcher"> <section class="info-dispatcher">
<div class="dispatcher" v-if="onlineScenery"> <div class="dispatcher" v-if="onlineScenery && onlineScenery.dispatcherExp != -1">
<span <span
class="dispatcher_level" class="dispatcher_level"
:style="calculateExpStyle(onlineScenery.dispatcherExp, onlineScenery.dispatcherIsSupporter)" :style="calculateExpStyle(onlineScenery.dispatcherExp, onlineScenery.dispatcherIsSupporter)"
@@ -42,14 +42,14 @@ import dateMixin from '../../../mixins/dateMixin';
import routerMixin from '../../../mixins/routerMixin'; import routerMixin from '../../../mixins/routerMixin';
import styleMixin from '../../../mixins/styleMixin'; import styleMixin from '../../../mixins/styleMixin';
import StationStatusBadge from '../../Global/StationStatusBadge.vue'; import StationStatusBadge from '../../Global/StationStatusBadge.vue';
import { OnlineScenery } from '../../../store/typings'; import { ActiveScenery } from '../../../store/typings';
import donatorMixin from '../../../mixins/donatorMixin'; import donatorMixin from '../../../mixins/donatorMixin';
export default defineComponent({ export default defineComponent({
mixins: [styleMixin, dateMixin, routerMixin, donatorMixin], mixins: [styleMixin, dateMixin, routerMixin, donatorMixin],
props: { props: {
onlineScenery: { onlineScenery: {
type: Object as PropType<OnlineScenery>, type: Object as PropType<ActiveScenery>,
required: false required: false
} }
}, },
@@ -30,12 +30,12 @@
<script lang="ts"> <script lang="ts">
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import { OnlineScenery } from '../../../store/typings'; import { ActiveScenery } from '../../../store/typings';
export default defineComponent({ export default defineComponent({
props: { props: {
onlineScenery: { onlineScenery: {
type: Object as PropType<OnlineScenery>, type: Object as PropType<ActiveScenery>,
required: false required: false
} }
}, },
@@ -32,14 +32,14 @@
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import modalTrainMixin from '../../../mixins/modalTrainMixin'; import modalTrainMixin from '../../../mixins/modalTrainMixin';
import routerMixin from '../../../mixins/routerMixin'; import routerMixin from '../../../mixins/routerMixin';
import { OnlineScenery } from '../../../store/typings'; import { ActiveScenery } from '../../../store/typings';
export default defineComponent({ export default defineComponent({
mixins: [routerMixin, modalTrainMixin], mixins: [routerMixin, modalTrainMixin],
props: { props: {
onlineScenery: { onlineScenery: {
type: Object as PropType<OnlineScenery>, type: Object as PropType<ActiveScenery>,
required: false required: false
} }
} }
+11 -12
View File
@@ -6,10 +6,10 @@
<span>{{ $t('scenery.timetables') }}</span> <span>{{ $t('scenery.timetables') }}</span>
<span> <span>
<span class="text--primary">{{ onlineScenery?.scheduledTrainCount.all || 0 }}</span> <span class="text--primary">{{ onlineScenery?.scheduledTrainCount.all ?? 0 }}</span>
<span> / </span> <span> / </span>
<span class="text--grayed"> <span class="text--grayed">
{{ onlineScenery?.scheduledTrainCount.confirmed || '0' }} {{ onlineScenery?.scheduledTrainCount.confirmed ?? 0 }}
</span> </span>
</span> </span>
@@ -33,12 +33,12 @@
{{ (i > 0 && '&bull;') || '' }} {{ (i > 0 && '&bull;') || '' }}
<button <button
:key="cp.checkpointName" :key="cp"
class="checkpoint_item" class="checkpoint_item"
:class="{ current: chosenCheckpoint === cp.checkpointName }" :class="{ current: chosenCheckpoint === cp }"
@click="setCheckpoint(cp)" @click="setCheckpoint(cp)"
> >
{{ cp.checkpointName }} {{ cp }}
</button> </button>
</span> </span>
</div> </div>
@@ -190,7 +190,7 @@ import Station from '../../scripts/interfaces/Station';
import { useMainStore } from '../../store/mainStore'; import { useMainStore } from '../../store/mainStore';
import modalTrainMixin from '../../mixins/modalTrainMixin'; import modalTrainMixin from '../../mixins/modalTrainMixin';
import ScheduledTrainStatus from './ScheduledTrainStatus.vue'; import ScheduledTrainStatus from './ScheduledTrainStatus.vue';
import { OnlineScenery } from '../../store/typings'; import { ActiveScenery } from '../../store/typings';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
export default defineComponent({ export default defineComponent({
@@ -205,7 +205,7 @@ export default defineComponent({
type: Object as PropType<Station> type: Object as PropType<Station>
}, },
onlineScenery: { onlineScenery: {
type: Object as PropType<OnlineScenery> type: Object as PropType<ActiveScenery>
} }
}, },
@@ -231,7 +231,7 @@ export default defineComponent({
const chosenCheckpoint = ref( const chosenCheckpoint = ref(
props.station?.generalInfo?.checkpoints?.length == 0 props.station?.generalInfo?.checkpoints?.length == 0
? '' ? ''
: props.station?.generalInfo?.checkpoints[0].checkpointName || null : props.station?.generalInfo?.checkpoints[0] ?? null
); );
return { return {
@@ -278,12 +278,11 @@ export default defineComponent({
loadSelectedOption() { loadSelectedOption() {
if (!this.station) return; if (!this.station) return;
this.chosenCheckpoint = this.chosenCheckpoint = this.station.generalInfo?.checkpoints[0] ?? this.station.name;
this.station.generalInfo?.checkpoints[0]?.checkpointName || this.station.name;
}, },
setCheckpoint(cp: { checkpointName: string }) { setCheckpoint(cp: string) {
this.chosenCheckpoint = cp.checkpointName; this.chosenCheckpoint = cp;
} }
} }
}); });
@@ -74,7 +74,7 @@ import dateMixin from '../../mixins/dateMixin';
import Station from '../../scripts/interfaces/Station'; import Station from '../../scripts/interfaces/Station';
import Loading from '../Global/Loading.vue'; import Loading from '../Global/Loading.vue';
import listObserverMixin from '../../mixins/listObserverMixin'; import listObserverMixin from '../../mixins/listObserverMixin';
import { OnlineScenery } from '../../store/typings'; import { ActiveScenery } from '../../store/typings';
import { API } from '../../typings/api'; import { API } from '../../typings/api';
import { Status } from '../../typings/common'; import { Status } from '../../typings/common';
import http from '../../http'; import http from '../../http';
@@ -87,7 +87,7 @@ export default defineComponent({
type: Object as PropType<Station> type: Object as PropType<Station>
}, },
onlineScenery: { onlineScenery: {
type: Object as PropType<OnlineScenery> type: Object as PropType<ActiveScenery>
} }
}, },
+1 -1
View File
@@ -132,7 +132,7 @@
<td class="station_dispatcher-exp"> <td class="station_dispatcher-exp">
<span <span
v-if="station.onlineInfo" v-if="station.onlineInfo && station.onlineInfo?.dispatcherExp != -1"
:style=" :style="
calculateExpStyle( calculateExpStyle(
station.onlineInfo.dispatcherExp, station.onlineInfo.dispatcherExp,
+2 -3
View File
@@ -42,14 +42,13 @@ export interface Filter {
nonPublic: boolean; nonPublic: boolean;
unavailable: boolean; unavailable: boolean;
abandoned: boolean; abandoned: boolean;
endingStatus: boolean; endingStatus: boolean;
afkStatus: boolean; afkStatus: boolean;
noSpaceStatus: boolean; noSpaceStatus: boolean;
unavailableStatus: boolean; unavailableStatus: boolean;
unsignedStatus: boolean; unsignedStatus: boolean;
authors: string; authors: string;
onlineFromHours: number; onlineFromHours: number;
withActiveTimetables: boolean;
withoutActiveTimetables: boolean;
} }
+1 -14
View File
@@ -181,7 +181,7 @@ export default defineComponent({
isSBL: /sbl/gi.test(stop.stopName), isSBL: /sbl/gi.test(stop.stopName),
position: stop.beginsHere ? 'begin' : stop.terminatesHere ? 'end' : 'en-route', position: stop.beginsHere ? 'begin' : stop.terminatesHere ? 'end' : 'en-route',
sceneryHash: '', sceneryHash: '',
sceneryName: this.timetableSceneryNames[currentSceneryIndex], sceneryName: this.train.timetableData!.sceneryNames[currentSceneryIndex],
status: stop.confirmed ? 'confirmed' : stop.stopped ? 'stopped' : 'unconfirmed' status: stop.confirmed ? 'confirmed' : stop.stopped ? 'stopped' : 'unconfirmed'
}; };
}) ?? [] }) ?? []
@@ -216,19 +216,6 @@ export default defineComponent({
} }
return activeMinorStopList; return activeMinorStopList;
},
timetableSceneryNames() {
if (!this.train.timetableData?.sceneries) return [];
return this.train.timetableData?.sceneries
.map(
(sceneryHash) =>
this.store.onlineSceneryList.find((st) => st.hash === sceneryHash)?.name ??
this.apiStore.sceneryData.find((sd) => sd.hash === sceneryHash)?.name ??
sceneryHash
)
.reverse();
} }
}, },
+16 -1
View File
@@ -7,7 +7,8 @@
"addons", "addons",
"blockades", "blockades",
"signals", "signals",
"status" "status",
"timetables"
], ],
"options": [ "options": [
@@ -228,6 +229,20 @@
"section": "status", "section": "status",
"value": true, "value": true,
"defaultValue": true "defaultValue": true
},
{
"id": "withActiveTimetables",
"name": "withActiveTimetables",
"section": "timetables",
"value": true,
"defaultValue": true
},
{
"id": "withoutActiveTimetables",
"name": "withoutActiveTimetables",
"section": "timetables",
"value": true,
"defaultValue": true
} }
], ],
"sliders": [ "sliders": [
+6 -1
View File
@@ -176,7 +176,8 @@
"signals": "SIGNALLING", "signals": "SIGNALLING",
"addons": "ADDITIONAL PROGRAMS", "addons": "ADDITIONAL PROGRAMS",
"blockades": "BLOCK SIGNALLING", "blockades": "BLOCK SIGNALLING",
"status": "ONLINE STATUS" "status": "ONLINE STATUS",
"timetables": "ACTIVE TIMETABLES"
}, },
"all-available": "ALL AVAILABLE", "all-available": "ALL AVAILABLE",
@@ -219,6 +220,10 @@
"historical": "HISTORICAL", "historical": "HISTORICAL",
"free": "FREE", "free": "FREE",
"occupied": "OCCUPIED", "occupied": "OCCUPIED",
"withActiveTimetables": "ACTIVE",
"withoutActiveTimetables": "NO ACTIVE",
"sliders": { "sliders": {
"min-lvl": "MIN. REQUIRED DISPATCHER LEVEL", "min-lvl": "MIN. REQUIRED DISPATCHER LEVEL",
"max-lvl": "MAX. REQUIRED DISPATCHER LEVEL", "max-lvl": "MAX. REQUIRED DISPATCHER LEVEL",
+5 -1
View File
@@ -165,7 +165,8 @@
"signals": "TYP SYGNALIZACJI", "signals": "TYP SYGNALIZACJI",
"addons": "DODATKOWE PROGRAMY", "addons": "DODATKOWE PROGRAMY",
"blockades": "BLOKADY LINIOWE", "blockades": "BLOKADY LINIOWE",
"status": "STATUS ONLINE" "status": "STATUS ONLINE",
"timetables": "AKTYWNE ROZKŁADY JAZDY"
}, },
"all-available": "WSZYSTKIE DOSTĘPNE", "all-available": "WSZYSTKIE DOSTĘPNE",
@@ -209,6 +210,9 @@
"free": "WOLNA", "free": "WOLNA",
"occupied": "ZAJĘTA", "occupied": "ZAJĘTA",
"withActiveTimetables": "AKTYWNE RJ",
"withoutActiveTimetables": "BEZ AKTYWNYCH RJ",
"sliders": { "sliders": {
"min-lvl": "MIN. WYMAGANY POZIOM DYŻURNEGO", "min-lvl": "MIN. WYMAGANY POZIOM DYŻURNEGO",
"max-lvl": "MAKS. WYMAGANY POZIOM DYŻURNEGO", "max-lvl": "MAKS. WYMAGANY POZIOM DYŻURNEGO",
+4 -6
View File
@@ -1,4 +1,4 @@
import { Availability, OnlineScenery, ScheduledTrain } from '../../store/typings'; import { Availability, ActiveScenery, ScheduledTrain } from '../../store/typings';
import { StationRoutes } from './StationRoutes'; import { StationRoutes } from './StationRoutes';
export default interface Station { export default interface Station {
@@ -8,6 +8,7 @@ export default interface Station {
name: string; name: string;
url: string; url: string;
abbr: string; abbr: string;
hash?: string;
reqLevel: number; reqLevel: number;
// supportersOnly: boolean; // supportersOnly: boolean;
@@ -25,11 +26,8 @@ export default interface Station {
availability: Availability; availability: Availability;
routes: StationRoutes; routes: StationRoutes;
checkpoints: { checkpoints: string[];
checkpointName: string;
scheduledTrains: ScheduledTrain[];
}[];
}; };
onlineInfo?: OnlineScenery; onlineInfo?: ActiveScenery;
} }
+1
View File
@@ -33,5 +33,6 @@ export default interface Train {
SKR: boolean; SKR: boolean;
routeDistance: number; routeDistance: number;
sceneries: string[]; sceneries: string[];
sceneryNames: string[];
}; };
} }
+28 -7
View File
@@ -93,26 +93,41 @@ export const sortStations = (
}; };
export const filterStations = (station: Station, filters: Filter) => { export const filterStations = (station: Station, filters: Filter) => {
if (!station.onlineInfo && filters['free']) return false; if (filters['free'] && (!station.onlineInfo || station.onlineInfo.dispatcherId == -1))
return false;
if (station.onlineInfo) { if (station.onlineInfo) {
const { dispatcherStatus } = station.onlineInfo; const { dispatcherStatus } = station.onlineInfo;
const isEnding = dispatcherStatus == Status.ActiveDispatcher.ENDING && filters['endingStatus']; const excludeEnding =
dispatcherStatus == Status.ActiveDispatcher.ENDING && filters['endingStatus'];
const isNotSigned = const excludeNotSigned =
(dispatcherStatus == Status.ActiveDispatcher.NOT_LOGGED_IN || (dispatcherStatus == Status.ActiveDispatcher.NOT_LOGGED_IN ||
dispatcherStatus == Status.ActiveDispatcher.UNAVAILABLE) && dispatcherStatus == Status.ActiveDispatcher.UNAVAILABLE) &&
filters['unavailableStatus']; filters['unavailableStatus'];
const isAFK = dispatcherStatus == Status.ActiveDispatcher.AFK && filters['afkStatus']; const excludeAFK = dispatcherStatus == Status.ActiveDispatcher.AFK && filters['afkStatus'];
const isNoSpace = const excludeNoSpace =
dispatcherStatus == Status.ActiveDispatcher.NO_SPACE && filters['noSpaceStatus']; dispatcherStatus == Status.ActiveDispatcher.NO_SPACE && filters['noSpaceStatus'];
const isOccupied = station.onlineInfo && filters['occupied']; const excludeOccupied = filters['occupied'] && dispatcherStatus != Status.ActiveDispatcher.FREE;
if (isEnding || isNotSigned || isAFK || isNoSpace || isOccupied) return false; const excludeActiveTTs =
(dispatcherStatus == Status.ActiveDispatcher.FREE ||
station.onlineInfo.scheduledTrainCount.all != 0) &&
filters['withActiveTimetables'];
if (
excludeEnding ||
excludeAFK ||
excludeNoSpace ||
excludeNotSigned ||
excludeOccupied ||
excludeActiveTTs
)
return false;
if ( if (
filters['onlineFromHours'] > 0 && filters['onlineFromHours'] > 0 &&
@@ -121,6 +136,12 @@ export const filterStations = (station: Station, filters: Filter) => {
return false; return false;
} }
const excludeNoActiveTTs =
filters['withoutActiveTimetables'] &&
(!station.onlineInfo || station.onlineInfo.scheduledTrainCount.all == 0);
if (excludeNoActiveTTs) return false;
if ( if (
(station.generalInfo?.availability == 'nonPublic' || !station.generalInfo) && (station.generalInfo?.availability == 'nonPublic' || !station.generalInfo) &&
filters['nonPublic'] filters['nonPublic']
+105 -35
View File
@@ -2,7 +2,7 @@ import { defineStore } from 'pinia';
import Train from '../scripts/interfaces/Train'; import Train from '../scripts/interfaces/Train';
import { parseSpawns, getScheduledTrains, getStationTrains } from './utils'; import { parseSpawns, getScheduledTrains, getStationTrains } from './utils';
import { OnlineScenery, ScheduledTrain, StoreState } from './typings'; import { ActiveScenery, ScheduledTrain, StoreState } from './typings';
import { Status } from '../typings/common'; import { Status } from '../typings/common';
import Station from '../scripts/interfaces/Station'; import Station from '../scripts/interfaces/Station';
@@ -41,6 +41,14 @@ export const useMainStore = defineStore('store', {
const timetable = train.timetable; const timetable = train.timetable;
const sceneryNames =
train.timetable?.sceneries?.map(
(sceneryHash) =>
this.activeSceneryList.find((st) => st.hash === sceneryHash)?.name ??
apiStore.sceneryData.find((sd) => sd.hash === sceneryHash)?.name ??
sceneryHash
) ?? [];
return { return {
trainId: train.driverName + train.trainNo.toString(), trainId: train.driverName + train.trainNo.toString(),
@@ -76,43 +84,67 @@ export const useMainStore = defineStore('store', {
category: timetable.category, category: timetable.category,
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()
} }
: undefined : undefined
} as Train; } as Train;
}); });
}, },
onlineSceneryList(state): OnlineScenery[] { activeSceneryList(state): ActiveScenery[] {
const apiStore = useApiStore(); const apiStore = useApiStore();
if (state.isOffline) return []; if (state.isOffline) return [];
if (!apiStore.activeData?.activeSceneries) return []; if (!apiStore.activeData?.activeSceneries) return [];
return apiStore.activeData?.activeSceneries.reduce((list, scenery) => { const offlineActiveSceneries = this.trainList.reduce((acc, train) => {
if (!train.timetableData) return acc;
train.timetableData.sceneryNames.forEach((name) => {
if (
acc.findIndex((v) => v.name == name && v.region == train.region) != -1 ||
apiStore.activeData?.activeSceneries?.findIndex(
(sc) => sc.stationName === name && sc.region == train.region
) != -1
)
return acc;
acc.push({
name: name,
hash: '',
region: train.region,
maxUsers: 0,
currentUsers: 0,
spawns: [],
dispatcherName: '',
dispatcherRate: 0,
dispatcherId: -1,
dispatcherExp: -1,
dispatcherIsSupporter: false,
scheduledTrains: [],
stationTrains: [],
dispatcherStatus: Status.ActiveDispatcher.FREE,
dispatcherTimestamp: -1,
isOnline: false,
scheduledTrainCount: {
all: 0,
confirmed: 0,
unconfirmed: 0
}
});
});
return acc;
}, [] as ActiveScenery[]);
const onlineActiveSceneries = apiStore.activeData?.activeSceneries.reduce((list, scenery) => {
if (scenery.isOnline !== 1 && Date.now() - scenery.lastSeen > 1000 * 60 * 2) return list; if (scenery.isOnline !== 1 && Date.now() - scenery.lastSeen > 1000 * 60 * 2) return list;
if (scenery.dispatcherStatus == Status.ActiveDispatcher.UNKNOWN) return list; if (scenery.dispatcherStatus == Status.ActiveDispatcher.UNKNOWN) return list;
const station = this.stationList.find((s) => s.name === scenery.stationName);
const scheduledTrains = getScheduledTrains(this.trainList, scenery, station?.generalInfo);
const stationTrains = getStationTrains(
this.trainList,
scheduledTrains,
this.region.id,
scenery
);
// Remove checkpoint duplicates
const uniqueScheduledTrains = scheduledTrains.reduce(
(uniqueList, sTrain) =>
uniqueList.find((v) => v.trainId === sTrain.trainId)
? uniqueList
: [...uniqueList, sTrain],
[] as ScheduledTrain[]
);
const dispatcherTimestamp = const dispatcherTimestamp =
scenery.dispatcherStatus == Status.ActiveDispatcher.NO_LIMIT scenery.dispatcherStatus == Status.ActiveDispatcher.NO_LIMIT
? Date.now() + 25500000 ? Date.now() + 25500000
@@ -132,22 +164,64 @@ export const useMainStore = defineStore('store', {
dispatcherId: scenery.dispatcherId, dispatcherId: scenery.dispatcherId,
dispatcherExp: scenery.dispatcherExp, dispatcherExp: scenery.dispatcherExp,
dispatcherIsSupporter: scenery.dispatcherIsSupporter, dispatcherIsSupporter: scenery.dispatcherIsSupporter,
scheduledTrains: scheduledTrains,
stationTrains: stationTrains,
dispatcherStatus: scenery.dispatcherStatus, dispatcherStatus: scenery.dispatcherStatus,
dispatcherTimestamp: dispatcherTimestamp, dispatcherTimestamp: dispatcherTimestamp,
isOnline: scenery.isOnline == 1, isOnline: scenery.isOnline == 1,
scheduledTrains: [],
stationTrains: [],
scheduledTrainCount: { scheduledTrainCount: {
all: uniqueScheduledTrains.length, all: 0,
confirmed: uniqueScheduledTrains.filter((train) => train.stopInfo.confirmed).length, confirmed: 0,
unconfirmed: uniqueScheduledTrains.filter((train) => !train.stopInfo.confirmed).length unconfirmed: 0
} }
}); });
return list; return list;
}, [] as OnlineScenery[]); }, [] as ActiveScenery[]);
const allActiveSceneries = [...onlineActiveSceneries, ...offlineActiveSceneries];
for (let i = 0, n = allActiveSceneries.length; i < n; i++) {
const scenery = allActiveSceneries[i];
const station = this.stationList.find((s) => s.name === scenery.name);
const scheduledTrains = getScheduledTrains(
this.trainList,
station?.generalInfo,
scenery.name,
scenery.region
);
const stationTrains = getStationTrains(
this.trainList,
scheduledTrains,
this.region.id,
scenery.name
);
// Remove checkpoint duplicates
const uniqueScheduledTrains = scheduledTrains.reduce(
(uniqueList, sTrain) =>
uniqueList.find((v) => v.trainId === sTrain.trainId)
? uniqueList
: [...uniqueList, sTrain],
[] as ScheduledTrain[]
);
scenery.scheduledTrains = scheduledTrains;
scenery.stationTrains = stationTrains;
scenery.scheduledTrainCount = {
all: uniqueScheduledTrains.length,
confirmed: uniqueScheduledTrains.filter((train) => train.stopInfo.confirmed).length,
unconfirmed: uniqueScheduledTrains.filter((train) => !train.stopInfo.confirmed).length
};
}
return allActiveSceneries;
}, },
stationList(): Station[] { stationList(): Station[] {
@@ -187,11 +261,7 @@ export const useMainStore = defineStore('store', {
...scenery, ...scenery,
authors: scenery.authors?.split(',').map((a) => a.trim()), authors: scenery.authors?.split(',').map((a) => a.trim()),
routes: routes, routes: routes,
checkpoints: scenery.checkpoints checkpoints: scenery.checkpoints?.split(';') ?? []
? scenery.checkpoints
.split(';')
.map((sub) => ({ checkpointName: sub, scheduledTrains: [] }))
: []
} }
}; };
}); });
+5 -2
View File
@@ -48,6 +48,8 @@ const filterInitStates: Filter = {
noSpaceStatus: false, noSpaceStatus: false,
unavailableStatus: false, unavailableStatus: false,
unsignedStatus: false, unsignedStatus: false,
withActiveTimetables: false,
withoutActiveTimetables: false,
authors: '', authors: '',
@@ -73,7 +75,7 @@ export const useStationFiltersStore = defineStore('stationFiltersStore', {
const store = useMainStore(); const store = useMainStore();
const savedStationNames = store.stationList.map((s) => s.name); const savedStationNames = store.stationList.map((s) => s.name);
const onlineUnsavedStations = store.onlineSceneryList const onlineUnsavedStations = store.activeSceneryList
.filter((os) => !savedStationNames.includes(os.name) && os.region == store.region.id) .filter((os) => !savedStationNames.includes(os.name) && os.region == store.region.id)
.map((os) => ({ .map((os) => ({
name: os.name, name: os.name,
@@ -85,7 +87,8 @@ export const useStationFiltersStore = defineStore('stationFiltersStore', {
...onlineUnsavedStations, ...onlineUnsavedStations,
...store.stationList.map((station) => ({ ...store.stationList.map((station) => ({
...station, ...station,
onlineInfo: store.onlineSceneryList.find( // append to 'onlineInfo' object for filtering legacy reasons - to optimize later (hopefully)
onlineInfo: store.activeSceneryList.find(
(os) => os.name == station.name && os.region == store.region.id (os) => os.name == station.name && os.region == store.region.id
) )
})) }))
+1 -1
View File
@@ -63,7 +63,7 @@ export interface StationJSONData {
availability: Availability; availability: Availability;
} }
export interface OnlineScenery { export interface ActiveScenery {
name: string; name: string;
hash: string; hash: string;
region: string; region: string;
+37 -84
View File
@@ -1,6 +1,5 @@
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 { API } from '../typings/api';
import { ScheduledTrain, StationTrain, StopStatus, TrainStop } from './typings'; import { ScheduledTrain, StationTrain, StopStatus, TrainStop } from './typings';
export function getLocoURL(locoType: string): string { export function getLocoURL(locoType: string): string {
@@ -102,51 +101,33 @@ export function getCheckpointTrain(
let prevStationName = '', let prevStationName = '',
nextStationName = ''; nextStationName = '';
let prevDepartureLine: string | null = null,
nextArrivalLine: string | null = null;
for (let i = trainStopIndex - 1; i >= 0; i--) {
if (/strong|podg/g.test(followingStops[i].stopName)) {
prevStationName = followingStops[i].stopNameRAW.replace(/,.*/g, '');
break;
}
}
for (let i = trainStopIndex + 1; i < followingStops.length; i++) {
if (/strong|podg/g.test(followingStops[i].stopName)) {
nextStationName = followingStops[i].stopNameRAW.replace(/,.*/g, '');
break;
}
}
let departureLine: string | null = null; let departureLine: string | null = null;
let arrivingLine: string | null = null; let arrivingLine: string | null = null;
for (let i = trainStopIndex; i < followingStops.length; i++) { let prevDepartureLine: string | null = null,
const currentStop = followingStops[i]; nextArrivalLine: string | null = null;
if (currentStop.departureLine == null) continue; for (let i = trainStopIndex; i >= 0; i--) {
const stop = followingStops[i];
if (!/-|_|it|sbl/gi.test(currentStop.departureLine)) { if (/strong|podg\.|pe\./g.test(stop.stopName) && !prevStationName && i <= trainStopIndex - 1)
departureLine = currentStop.departureLine; prevStationName = stop.stopNameRAW.replace(/,.*/g, '');
nextArrivalLine = followingStops[i + 1]?.arrivalLine || null;
break; if (stop.arrivalLine != null && !arrivingLine && !/-|_|it|sbl/gi.test(stop.arrivalLine)) {
arrivingLine = stop.arrivalLine;
prevDepartureLine = followingStops[i - 1]?.departureLine || null;
} }
} }
for (let i = trainStopIndex; i >= 0; i--) { for (let i = trainStopIndex; i < followingStops.length; i++) {
const currentStop = followingStops[i]; const stop = followingStops[i];
if (currentStop.arrivalLine == null) continue; if (/strong|podg\.|pe\./g.test(stop.stopName) && !nextStationName && i > trainStopIndex)
nextStationName = stop.stopNameRAW.replace(/,.*/g, '');
if (!/-|_|it|sbl/gi.test(currentStop.arrivalLine)) { if (stop.departureLine && !departureLine && !/-|_|it|sbl/gi.test(stop.departureLine)) {
arrivingLine = currentStop.arrivalLine; departureLine = stop.departureLine;
prevDepartureLine = followingStops[i - 1]?.departureLine || null; nextArrivalLine = followingStops[i + 1]?.arrivalLine || null;
break;
} }
} }
@@ -177,8 +158,8 @@ export function getCheckpointTrain(
region: train.region, region: train.region,
arrivingLine, arrivingLine: arrivingLine,
departureLine, departureLine: departureLine,
nextArrivalLine, nextArrivalLine,
prevDepartureLine prevDepartureLine
@@ -187,59 +168,33 @@ export function getCheckpointTrain(
export function getScheduledTrains( export function getScheduledTrains(
trainList: Train[], trainList: Train[],
sceneryData: API.ActiveSceneries.Data, stationGeneralInfo: Station['generalInfo'],
stationGeneralInfo: Station['generalInfo'] stationName: string,
region: string
// sceneryData: API.ActiveSceneries.Data,
): ScheduledTrain[] { ): ScheduledTrain[] {
const stationNameLower = sceneryData.stationName.toLocaleLowerCase(); // stationGeneralInfo?.checkpoints.forEach((cp) => (cp.scheduledTrains.length = 0));
stationGeneralInfo?.checkpoints.forEach((cp) => (cp.scheduledTrains.length = 0));
return trainList.reduce((acc: ScheduledTrain[], train) => { return trainList.reduce((acc: ScheduledTrain[], train) => {
if (!train.timetableData) return acc; if (!train.timetableData) return acc;
if (train.region != sceneryData.region) return acc; if (train.region != region) return acc;
const timetable = train.timetableData; const timetable = train.timetableData;
if (!timetable.sceneries.includes(sceneryData.stationHash)) return acc; if (!timetable.sceneryNames.includes(stationName)) return acc;
const stopInfoIndex = timetable.followingStops.findIndex((stop) => { const checkpoints = [stationName];
const stopNameLower = stop.stopNameRAW.toLocaleLowerCase(); if (stationGeneralInfo?.checkpoints) checkpoints.push(...stationGeneralInfo.checkpoints);
return (
stationNameLower == stopNameLower ||
(!/(po\.|podg\.)/.test(stopNameLower) && stopNameLower.includes(stationNameLower)) ||
(!/(po\.|podg\.)/.test(stationNameLower) && stationNameLower.includes(stopNameLower)) ||
(stopNameLower.split(', podg.')[0] !== undefined &&
stationNameLower.startsWith(stopNameLower.split(', podg.')[0]))
);
});
const checkpointScheduledTrains: ScheduledTrain[] = []; const checkpointScheduledTrains: ScheduledTrain[] = [];
for (let i = 0; i < timetable.followingStops.length; i++) {
if (stopInfoIndex != -1) {
checkpointScheduledTrains.push(
getCheckpointTrain(train, stopInfoIndex, sceneryData.stationName)
);
}
stationGeneralInfo?.checkpoints?.forEach((checkpoint) => {
// if (checkpoint.checkpointName.toLocaleLowerCase() == stationNameLower) return;
if ( if (
checkpointScheduledTrains.findIndex( new RegExp(`^(${checkpoints.join('|')})$`, 'i').test(
(cpTrain) => timetable.followingStops[i].stopNameRAW
cpTrain.checkpointName.toLocaleLowerCase() == )
checkpoint.checkpointName.toLocaleLowerCase() ) {
) != -1 checkpointScheduledTrains.push(getCheckpointTrain(train, i, stationName));
) }
return; }
const index = timetable.followingStops.findIndex(
(stop) => stop.stopNameRAW.toLowerCase() == checkpoint.checkpointName.toLowerCase()
);
if (index > -1)
checkpointScheduledTrains.push(getCheckpointTrain(train, index, sceneryData.stationName));
});
acc.push(...checkpointScheduledTrains); acc.push(...checkpointScheduledTrains);
return acc; return acc;
@@ -250,14 +205,12 @@ export function getStationTrains(
trainList: Train[], trainList: Train[],
scheduledTrainList: ScheduledTrain[], scheduledTrainList: ScheduledTrain[],
region: string, region: string,
sceneryData: API.ActiveSceneries.Data stationName: string
): StationTrain[] { ): StationTrain[] {
return trainList return trainList
.filter( .filter(
(train) => (train) =>
train?.region === region && train?.region === region && train.online && train.currentStationName === stationName
train.online &&
train.currentStationName === sceneryData.stationName
) )
.map((train) => ({ .map((train) => ({
driverName: train.driverName, driverName: train.driverName,
+1
View File
@@ -1,5 +1,6 @@
export namespace Status { export namespace Status {
export enum ActiveDispatcher { export enum ActiveDispatcher {
FREE = -3,
INVALID = -2, INVALID = -2,
UNKNOWN = -1, UNKNOWN = -1,
NO_LIMIT = 0, NO_LIMIT = 0,
+2 -6
View File
@@ -147,7 +147,7 @@ export default defineComponent({
}, },
onlineSceneryInfo() { onlineSceneryInfo() {
return this.store.onlineSceneryList.find( return this.store.activeSceneryList.find(
(scenery) => (scenery) =>
scenery.name === this.station?.toString().replace(/_/g, ' ') && scenery.name === this.station?.toString().replace(/_/g, ' ') &&
scenery.region == this.store.region.id scenery.region == this.store.region.id
@@ -169,11 +169,7 @@ export default defineComponent({
loadSelectedCheckpoint() { loadSelectedCheckpoint() {
if (!this.stationInfo?.generalInfo?.checkpoints) return; if (!this.stationInfo?.generalInfo?.checkpoints) return;
if (this.stationInfo.generalInfo.checkpoints.length == 0) return; if (this.stationInfo.generalInfo.checkpoints.length == 0) return;
this.selectedCheckpoint = this.stationInfo.generalInfo.checkpoints[0].checkpointName; this.selectedCheckpoint = this.stationInfo.generalInfo.checkpoints[0];
},
selectCheckpoint(cp: { checkpointName: string }) {
this.selectedCheckpoint = cp.checkpointName;
} }
} }
}); });