Compare commits

..

9 Commits

Author SHA1 Message Date
Spythere 83444f64d0 Merge pull request #155 from Spythere/development
v1.32.0
2026-02-26 21:15:04 +01:00
Spythere c2f7eef146 Merge pull request #153 from Spythere/development
v1.31.1
2026-01-16 22:27:16 +01:00
Spythere 3c78af4dc0 Merge pull request #151 from Spythere/development
v1.31.0
2026-01-10 21:22:48 +01:00
Spythere fc7a9be9dd Merge pull request #150 from Spythere/development
Fix for station statistics dropdown overflow
2025-12-13 02:21:19 +01:00
Spythere 3c3a114a38 Merge pull request #149 from Spythere/development
Information about migration to the new domain
2025-12-13 00:20:13 +01:00
Spythere fe6972c1f8 Merge pull request #148 from Spythere/development
Extended isChristmas check from 20th to 6th December
2025-12-05 21:28:04 +01:00
Spythere 08b9b72dcd Merge pull request #147 from Spythere/development
Hotfix for VPS deploy
2025-12-04 00:27:38 +01:00
Spythere c90be042e7 Merge pull request #146 from Spythere/development
Updated GitHub workflow for deploying files to dedicated VPS
2025-12-04 00:21:41 +01:00
Spythere 430a05ab38 Merge pull request #145 from Spythere/development
v1.30.7
2025-11-28 01:14:13 +01:00
83 changed files with 3060 additions and 3949 deletions
+2 -1
View File
@@ -15,12 +15,13 @@ pnpm-debug.log*
# Editor directories and files # Editor directories and files
.idea .idea
.vscode
*.suo *.suo
*.ntvs* *.ntvs*
*.njsproj *.njsproj
*.sln *.sln
*.sw? *.sw?
.vscode/settings.json node_modules
*.log *.log
-7
View File
@@ -1,7 +0,0 @@
{
"$schema": "https://json.schemastore.org/prettierrc",
"tabWidth": 2,
"singleQuote": true,
"printWidth": 100,
"trailingComma": "none"
}
-3
View File
@@ -1,3 +0,0 @@
{
"recommendations": ["Vue.volar", "esbenp.prettier-vscode"]
}
+1 -1
View File
@@ -1,4 +1,4 @@
# [STACJOWNIK TD2](https://stacjownik-td2.spythere.eu) # [STACJOWNIK TD2](https://stacjownik-td2.web.app)
ODŚWIEŻANA LISTA SCENERII I SKŁADÓW ONLINE DLA [SYMULATORA TRAIN DRIVER 2](https://td2.info.pl) ODŚWIEŻANA LISTA SCENERII I SKŁADÓW ONLINE DLA [SYMULATORA TRAIN DRIVER 2](https://td2.info.pl)
Vendored
-1
View File
@@ -1 +0,0 @@
/// <reference types="vite/client" />
+3 -9
View File
@@ -20,7 +20,7 @@
<link rel="mask-icon" href="/safari-pinned-tab.svg" color="#5bbad5" /> <link rel="mask-icon" href="/safari-pinned-tab.svg" color="#5bbad5" />
<meta name="msapplication-TileColor" content="#da532c" /> <meta name="msapplication-TileColor" content="#da532c" />
<link rel="icon" href="/favicon.ico" /> <link rel="icon" href="favicon.ico" />
<link rel="stylesheet" href="/fa/css/fontawesome.css" /> <link rel="stylesheet" href="/fa/css/fontawesome.css" />
<link rel="stylesheet" href="/fa/css/brands.css" /> <link rel="stylesheet" href="/fa/css/brands.css" />
@@ -28,13 +28,7 @@
<link rel="stylesheet" href="/fa/css/solid.css" /> <link rel="stylesheet" href="/fa/css/solid.css" />
<!-- Preloads --> <!-- Preloads -->
<link <link rel="preload" href="fonts/Quicksand-Bold.woff2" as="font" type="font/woff2" crossorigin />
rel="preload"
href="/fonts/Quicksand-Bold.woff2"
as="font"
type="font/woff2"
crossorigin
/>
<link <link
rel="preload" rel="preload"
@@ -89,7 +83,7 @@
<!-- Static OpenGraph meta --> <!-- Static OpenGraph meta -->
<meta name="description" content="Pomocnik maszynisty i dyżurnego symulatora Train Driver 2" /> <meta name="description" content="Pomocnik maszynisty i dyżurnego symulatora Train Driver 2" />
<meta property="og:url" content="https://stacjownik-td2.spythere.eu/" /> <meta property="og:url" content="https://stacjownik-td2.web.app/" />
<meta property="og:type" content="website" /> <meta property="og:type" content="website" />
<meta property="og:title" content="Stacjownik" /> <meta property="og:title" content="Stacjownik" />
<meta <meta
+3 -3
View File
@@ -1,6 +1,6 @@
{ {
"name": "stacjownik", "name": "stacjownik",
"version": "1.35.0", "version": "1.32.0",
"private": true, "private": true,
"type": "module", "type": "module",
"scripts": { "scripts": {
@@ -26,12 +26,12 @@
"vue-router": "^4.4.0" "vue-router": "^4.4.0"
}, },
"devDependencies": { "devDependencies": {
"@tsconfig/node24": "^24.0.4", "@types/node": "^24.3.1",
"@types/node": "^24.12.0",
"@types/showdown": "^2.0.6", "@types/showdown": "^2.0.6",
"@vite-pwa/assets-generator": "^1.0.0", "@vite-pwa/assets-generator": "^1.0.0",
"@vitejs/plugin-vue": "^6.0.1", "@vitejs/plugin-vue": "^6.0.1",
"@vue/tsconfig": "^0.8.1", "@vue/tsconfig": "^0.8.1",
"axios": "^1.9.0",
"prettier": "^3.3.3", "prettier": "^3.3.3",
"typescript": "^5.5.4", "typescript": "^5.5.4",
"vite": "^7.1.4", "vite": "^7.1.4",
Binary file not shown.

Before

Width:  |  Height:  |  Size: 22 KiB

+24 -9
View File
@@ -30,6 +30,7 @@
<script lang="ts"> <script lang="ts">
import { defineComponent } from 'vue'; import { defineComponent } from 'vue';
import axios from 'axios';
import { version } from '../package.json'; import { version } from '../package.json';
import { Status } from './typings/common'; import { Status } from './typings/common';
@@ -49,6 +50,7 @@ import AppWelcomeCard from './components/App/AppWelcomeCard.vue';
const STORAGE_VERSION_KEY = 'app_version'; const STORAGE_VERSION_KEY = 'app_version';
const WELCOME_CARD_SEEN_KEY = 'welcome_card_seen'; const WELCOME_CARD_SEEN_KEY = 'welcome_card_seen';
const MIGRATE_INFO_CARD_SEEN_KEY = 'migrate_info_card_seen';
export default defineComponent({ export default defineComponent({
components: { components: {
@@ -90,6 +92,7 @@ export default defineComponent({
this.setupOfflineHandling(); this.setupOfflineHandling();
this.checkAppVersion(); this.checkAppVersion();
this.handleQueries(); this.handleQueries();
this.handleMigrateInfo();
this.apiStore.setupAPIData(); this.apiStore.setupAPIData();
}, },
@@ -100,6 +103,10 @@ export default defineComponent({
if (query.get('welcomeCard') == '1') { if (query.get('welcomeCard') == '1') {
this.isWelcomeCardOpen = true; this.isWelcomeCardOpen = true;
} }
if (query.get('migrateCard') == '1') {
this.store.isMigrateInfoCardOpen = true;
}
}, },
async checkAppVersion() { async checkAppVersion() {
@@ -113,15 +120,11 @@ export default defineComponent({
} }
try { try {
const response = await fetch( const releaseData = await (
'https://api.github.com/repos/Spythere/stacjownik/releases/latest' await axios.get('https://api.github.com/repos/Spythere/stacjownik/releases/latest')
); ).data;
if (!response.ok) { if (!releaseData) return;
throw new Error('Failed to fetch release data from repository!');
}
const releaseData = await response.json();
this.store.appUpdate = { this.store.appUpdate = {
version, version,
@@ -133,7 +136,7 @@ export default defineComponent({
(storageVersion != '' && storageVersion != version && this.isOnProductionHost) || (storageVersion != '' && storageVersion != version && this.isOnProductionHost) ||
import.meta.env.VITE_UPDATE_TEST === 'test'; import.meta.env.VITE_UPDATE_TEST === 'test';
} catch (error) { } catch (error) {
console.error(error); console.error(`Wystąpił błąd podczas pobierania danych z API GitHuba: ${error}`);
} }
StorageManager.setStringValue(STORAGE_VERSION_KEY, version); StorageManager.setStringValue(STORAGE_VERSION_KEY, version);
@@ -162,6 +165,13 @@ export default defineComponent({
this.apiStore.connectToAPI(); this.apiStore.connectToAPI();
}, },
handleMigrateInfo() {
if (location.hostname != 'stacjownik-td2.web.app') return;
if (StorageManager.getBooleanValue(MIGRATE_INFO_CARD_SEEN_KEY) === true) return;
this.store.isMigrateInfoCardOpen = true;
},
loadLang() { loadLang() {
const storageLang = StorageManager.getStringValue('lang'); const storageLang = StorageManager.getStringValue('lang');
@@ -183,6 +193,11 @@ export default defineComponent({
closeWelcomeCard() { closeWelcomeCard() {
this.isWelcomeCardOpen = false; this.isWelcomeCardOpen = false;
StorageManager.setBooleanValue(WELCOME_CARD_SEEN_KEY, true); StorageManager.setBooleanValue(WELCOME_CARD_SEEN_KEY, true);
},
closeMigrateInfoCard() {
this.store.isMigrateInfoCardOpen = false;
StorageManager.setBooleanValue(MIGRATE_INFO_CARD_SEEN_KEY, true);
} }
} }
}); });
+3 -3
View File
@@ -63,19 +63,19 @@
</b> </b>
<div class="apps-grid"> <div class="apps-grid">
<a class="app-item" href="https://pojazdownik-td2.spythere.eu/" target="_blank"> <a class="app-item" href="https://pojazdownik-td2.web.app/" target="_blank">
<img src="/images/icon-pojazdownik.svg" alt="pojazdownik app logo" /> <img src="/images/icon-pojazdownik.svg" alt="pojazdownik app logo" />
<h3 class="text--primary">Pojazdownik</h3> <h3 class="text--primary">Pojazdownik</h3>
<p>{{ $t('welcome.pojazdownik-desc') }}</p> <p>{{ $t('welcome.pojazdownik-desc') }}</p>
</a> </a>
<a class="app-item" href="https://generator-td2.spythere.eu/" target="_blank"> <a class="app-item" href="https://generator-td2.web.app/" target="_blank">
<img src="/images/icon-gnr.svg" alt="generator app logo" /> <img src="/images/icon-gnr.svg" alt="generator app logo" />
<h3 class="text--primary">GeneraTOR</h3> <h3 class="text--primary">GeneraTOR</h3>
<p>{{ $t('welcome.generator-desc') }}</p> <p>{{ $t('welcome.generator-desc') }}</p>
</a> </a>
<a class="app-item" href="https://srjp-td2.spythere.eu/" target="_blank"> <a class="app-item" href="https://srjp-td2.web.app/" target="_blank">
<img src="/images/icon-srjp.svg" alt="srjp app logo" /> <img src="/images/icon-srjp.svg" alt="srjp app logo" />
<h3 class="text--primary">Rozkładownik</h3> <h3 class="text--primary">Rozkładownik</h3>
<p>{{ $t('welcome.srjp-desc') }}</p> <p>{{ $t('welcome.srjp-desc') }}</p>
+94
View File
@@ -0,0 +1,94 @@
<template>
<Card :is-open="isOpen" @toggle-card="toggleCard">
<div class="body-content">
<div class="content-top">
<img src="/images/icon-loading.svg" alt="loading" height="125" />
<h1>{{ t('migrate-info.header-text') }}</h1>
</div>
<div>
<p v-html="t('migrate-info.paragraph-1-html')"></p>
<p>
<a class="new-link" href="https://stacjownik-td2.spythere.eu/" target="_blank">
{{ t('migrate-info.paragraph-2-link-text') }}
</a>
</p>
<p>
{{ t('migrate-info.paragraph-3-text') }}
</p>
<p class="info-bottom" v-html="t('migrate-info.paragraph-4-html')"></p>
</div>
<div class="content-actions">
<button class="btn btn--action" @click="toggleCard">PRZYJĄŁEM!</button>
</div>
</div>
</Card>
</template>
<script lang="ts" setup>
import { useI18n } from 'vue-i18n';
import Card from '../Global/Card.vue';
const { t } = useI18n();
defineProps({
isOpen: {
type: Boolean,
required: true
}
});
const emit = defineEmits(['toggleCard']);
function toggleCard() {
emit('toggleCard');
}
</script>
<style lang="scss" scoped>
.body-content {
max-width: 800px;
min-height: 500px;
padding: 1em 0.5em;
display: grid;
grid-template-rows: auto 1fr auto;
gap: 0.5em;
text-align: center;
font-size: 1.1em;
}
p {
padding: 0.5em 0;
}
a.new-link {
font-size: 1.2em;
color: var(--clr-primary);
color: transparent;
background: var(--clr-primary);
background: linear-gradient(90deg, var(--clr-primary), #ffffff);
background-clip: text;
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
text-shadow: var(--clr-primary) 0 0 10px;
}
.info-bottom {
font-size: 0.9em;
color: #ccc;
}
.content-actions {
display: flex;
justify-content: center;
font-size: 1.2em;
}
</style>
+6 -30
View File
@@ -1,9 +1,7 @@
<template> <template>
<Card :is-open="isUpdateCardOpen" @toggle-card="toggleCard(false)"> <Card :is-open="isUpdateCardOpen" @toggle-card="toggleCard(false)">
<div class="content" tabindex="0" ref="content"> <div class="content" tabindex="0" ref="content">
<h1 class="content-title"> <h1 style="margin-bottom: 0.5em">🚀 {{ $t('update.title') }}</h1>
<i class="fa-solid fa-wand-sparkles"></i> {{ $t('update.title') }}
</h1>
<div class="features-body" v-if="htmlChangelog != ''" v-html="htmlChangelog"></div> <div class="features-body" v-if="htmlChangelog != ''" v-html="htmlChangelog"></div>
<div class="no-features" v-else>{{ $t('update.no-data') }}</div> <div class="no-features" v-else>{{ $t('update.no-data') }}</div>
@@ -18,7 +16,7 @@
<i18n-t keypath="update.info-2"> <i18n-t keypath="update.info-2">
<template v-slot:link> <template v-slot:link>
<a href="https://github.com/Spythere/stacjownik/releases" target="_blank">{{ <a href="https://github.com/Spythere/stacjownik" target="_blank">{{
$t('update.info-2-link-text') $t('update.info-2-link-text')
}}</a> }}</a>
</template> </template>
@@ -88,28 +86,19 @@ export default defineComponent({
} }
::v-deep(h2) { ::v-deep(h2) {
margin-top: 1em;
padding: 0.5em 0; padding: 0.5em 0;
border-bottom: 1px solid #aaa;
&::after {
content: '';
display: block;
height: 2px;
width: 100%;
background-color: #aaa;
margin-top: 0.25em;
}
} }
::v-deep(h3) { ::v-deep(h3) {
padding-bottom: 0.25em; padding: 0.5em 0;
} }
::v-deep(ul) { ::v-deep(ul) {
list-style: disc; list-style: disc;
line-height: 1.5em;
padding: 0 1.5em; padding: 0 1.5em;
line-height: 1.5em;
padding-bottom: 0.5em;
} }
.content { .content {
@@ -123,19 +112,6 @@ export default defineComponent({
max-width: 700px; max-width: 700px;
} }
.content-title {
color: var(--clr-primary);
color: transparent;
background: var(--clr-primary);
background: linear-gradient(90deg, var(--clr-primary) 30%, #ffffff 90%);
background-clip: text;
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
text-shadow: var(--clr-primary) 0 0 10px;
}
.no-features { .no-features {
text-align: center; text-align: center;
} }
@@ -1,325 +0,0 @@
<template>
<div class="driver-propositions">
<h3>{{ t('trains.number-propositions-header') }}</h3>
<div class="categories-select">
<button
v-for="(category, i) in availableCategories"
class="btn btn--option btn--action"
@click="selectCategory(i)"
:class="{ checked: i == chosenCategoryIndex }"
>
{{ category }}
</button>
</div>
<div v-if="numberPropositions.length > 0" class="propositions-numbers">
<div v-if="chosenCategory">
<b>{{ chosenCategory }} </b> -
{{ t(`categories.${chosenCategory.slice(0, 2)}`) }}
({{ t(`categories.${chosenCategory.slice(2)}`) }})
</div>
<div v-if="chosenCategoryRules">
<span v-if="chosenCategoryRules[0]"
>{{ t('trains.number-propositions-third-number') }}
<b class="text--primary">{{ chosenCategoryRules[0] }}</b> &bull;
</span>
<span
>{{
t('trains.number-propositions-last-nums', {
count: chosenCategoryRules[1].length
})
}}
<b class="text--primary">{{ chosenCategoryRules[1] }}</b> -
<b class="text--primary">{{ chosenCategoryRules[2] }}</b></span
>
</div>
<div style="margin-top: 0.5em">
<b>{{ t('trains.number-propositions-title') }}&nbsp;</b>
<i>{{ numberPropositions.join(', ') }}</i>
</div>
</div>
<div class="no-propositions" v-else>{{ t('trains.number-propositions-empty') }}</div>
<div class="cargo-warnings" v-if="getCargoWarnings.size > 0">
<hr />
<h3>{{ t('cargo-warnings.title') }}</h3>
<div class="warnings-container">
<div
v-for="warning in getCargoWarnings"
class="train-badge"
:class="`${warning.split('-')[0]}`"
>
{{ t('cargo-warnings.' + warning) }}
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { ref, computed, PropType, watch, onMounted } from 'vue';
import { useI18n } from 'vue-i18n';
import { Train } from '../../typings/common';
import rulesJSON from '../../data/trainNumberRules.json';
import { useApiStore } from '../../store/apiStore';
const { t } = useI18n();
const apiStore = useApiStore();
const props = defineProps({
chosenTrain: {
type: Object as PropType<Train>,
required: true
}
});
const emits = defineEmits(['selectCategory']);
const chosenCategoryIndex = ref(0);
const numberPropositions = ref<string[]>([]);
const chosenCategoryRules = ref<any[]>([]);
watch(
computed(() => props.chosenTrain.trainNo),
() => {
chosenCategoryIndex.value = 0;
generateNumberPropositions();
}
);
onMounted(() => {
generateNumberPropositions();
});
function generateNumberPropositions() {
const categoryCode = chosenCategory.value?.slice(0, 2);
const trainNoStr = props.chosenTrain.trainNo.toString();
// Get category rules
const rules = categoryCode
? ((rulesJSON.categoriesRules as any)[categoryCode] as any[])
: undefined;
if (!categoryCode || !rules) {
numberPropositions.value.length = 0;
chosenCategoryRules.value.length = 0;
return;
}
const [thirdNumber, minRange, maxRange] = rules;
const propositionsArr: string[] = [];
for (let i = 0; i < 5; i++) {
let generatedNumStr = '';
generatedNumStr += trainNoStr[0] ?? Math.floor(Math.random() * 10);
generatedNumStr += trainNoStr[1] ?? Math.floor(Math.random() * 10);
// Third number
generatedNumStr += thirdNumber ?? '';
// Remaining numbers
const rangeNums = minRange?.length ?? 3;
const randRange = Math.floor(
Math.random() * (Number(maxRange) - Number(minRange)) + Number(minRange)
).toString();
const leadingZeros = new Array(Math.abs(randRange.toString().length - rangeNums))
.fill('0')
.join('');
generatedNumStr += `${leadingZeros}${randRange}`;
const isNumberTaken =
apiStore.activeData?.trains?.some((t) => t.trainNo.toString() == generatedNumStr) ?? false;
if (!isNumberTaken) {
propositionsArr.push(generatedNumStr);
} else {
i--;
}
if (Number(randRange) > Number(maxRange)) break;
}
numberPropositions.value = propositionsArr;
chosenCategoryRules.value = rules;
}
const chosenCategory = computed(() => {
return availableCategories.value[chosenCategoryIndex.value];
});
const getCargoWarnings = computed(() => {
const stockList = props.chosenTrain.stockList;
let warnings: Set<string> = new Set();
stockList.forEach((stockVehicle) => {
const [vehicleName, vehicleCargo] = stockVehicle.split(':');
if (vehicleName.startsWith('WB117')) warnings.add(vehicleCargo ? 'twr-un1965' : 'tn-un1965');
else if (vehicleName.startsWith('445Rb'))
warnings.add(vehicleCargo ? 'tn-un1202' : 'tn-un1202-empty');
else if (vehicleName.startsWith('EDK80')) warnings.add('pn-edk80');
if (vehicleCargo) {
if (vehicleCargo.startsWith('wt_20')) warnings.add('pn-innofreight');
else if (/^(tank|vehicles_01|truck)/.test(vehicleCargo)) warnings.add('pn-military');
}
});
return warnings;
});
const availableCategories = computed(() => {
const stockList = props.chosenTrain.stockList;
const headVehicle = stockList[0]?.split('-')[0] ?? '';
let availableCategories: string[] = [];
let categoryTraction = 'E';
let vehicleTypesSet = new Set<string>();
let wagonsNamesSet = new Set<string>();
let cargoNamesSet = new Set<string>();
for (const stockName of stockList) {
const [vehicleName, ...cargoList] = stockName.split(':');
const vehicleData = apiStore.vehiclesData?.vehicles.find((v) => v.name == vehicleName);
if (!vehicleData) continue;
vehicleTypesSet.add(vehicleData.type);
if (vehicleData.type.startsWith('wagon-')) wagonsNamesSet.add(vehicleData.name.split('_')[0]);
if (cargoList !== undefined) cargoList.forEach((c) => cargoNamesSet.add(c.split('_')[0]));
}
let vehicleTypesArr = [...vehicleTypesSet];
let wagonsNamesArr = [...wagonsNamesSet];
// Traction
if (vehicleTypesArr[0] == 'loco-electric') categoryTraction = 'E';
else if (vehicleTypesArr[0] == 'loco-diesel') categoryTraction = 'S';
else if (vehicleTypesArr[0] == 'unit-electric') categoryTraction = 'J';
else categoryTraction = 'M';
// EMU / DMU - M*, R*, P*
if (vehicleTypesArr.length == 1 && (categoryTraction == 'J' || categoryTraction == 'M')) {
availableCategories.push('MO', 'MP', 'MM', 'RO', 'RP', 'RA', 'RM', 'PW');
}
// Only locos (up to 3) - LT, LP, LS
else if (stockList.length <= 3 && vehicleTypesArr.every((v) => v.startsWith('loco-'))) {
if (/^(EU|ET|201E|4E|SU|ST|M62|CTLR4C)/.test(headVehicle)) availableCategories.push('LT');
if (/^(EU|EP|SU|SP)/.test(headVehicle)) availableCategories.push('LP');
if (/^(SM)/.test(headVehicle)) availableCategories.push('LS');
}
// Only locos (more than 3) - TH
else if (stockList.length > 3 && vehicleTypesArr.every((v) => v.startsWith('loco-'))) {
availableCategories.push('TH');
}
// Loco(s) + passenger only wagons - M*, R*, E*, P*
else if (vehicleTypesArr.every((v) => v.startsWith('loco-') || v == 'wagon-passenger')) {
availableCategories.push('EI', 'EC', 'EN', 'MO', 'MP', 'MM', 'RO', 'RP', 'RA', 'RM', 'PW');
}
// Loco(s) + cargo only / mixed wagons - T*, Z*
else {
if (wagonsNamesArr.every((v) => /^(627Z|412Z)/.test(v)))
availableCategories.push('TC', 'TD', 'TS');
else if (stockList.slice(1).every((v) => /PKPE/.test(v))) {
availableCategories.push('ZU', 'ZN');
} else if (wagonsNamesArr.length < 3 || cargoNamesSet.size < 3) {
availableCategories.push('TM', 'TG', 'TS', 'TK');
} else {
availableCategories.push('TN', 'TR', 'TS', 'TK');
}
}
return availableCategories.map((c) => `${c}${categoryTraction}`);
});
function selectCategory(i: number) {
chosenCategoryIndex.value = i;
generateNumberPropositions();
}
</script>
<style lang="scss" scoped>
@use '../../styles/responsive';
@use '../../styles/badge';
.driver-propositions {
margin-bottom: 1em;
padding: 0.5em;
background-color: #111;
}
.categories-select {
display: inline-flex;
flex-wrap: wrap;
gap: 0.5em;
margin-top: 0.5em;
position: relative;
&::after {
content: '';
position: absolute;
bottom: calc(-0.5em);
left: 0;
width: 100%;
height: 2px;
background-color: #aaa;
}
}
.propositions-numbers {
margin-top: 1em;
}
.no-propositions {
margin-top: 1em;
color: #ccc;
}
.cargo-warnings {
margin-top: 0.5em;
h3 {
margin: 0.5em 0;
}
}
.warnings-container {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
}
@include responsive.smallScreen {
.driver-propositions {
text-align: center;
}
.categories-select {
justify-content: center;
}
.warnings-container {
justify-content: center;
}
}
</style>
@@ -13,7 +13,7 @@
<div class="actions actions-right"> <div class="actions actions-right">
<a <a
class="a-button btn--filled btn--image" class="a-button btn--filled btn--image"
:href="`https://srjp-td2.spythere.eu/?id=${chosenTrain.id}`" :href="`https://srjp-td2.web.app/?id=${chosenTrain.id}`"
target="_blank" target="_blank"
> >
<span class="hidable"> <span class="hidable">
+245 -8
View File
@@ -4,21 +4,66 @@
<!-- Train action buttons --> <!-- Train action buttons -->
<div class="train-stock-actions"> <div class="train-stock-actions">
<button class="btn btn--action" @click="copyStockToClipboard()"> <button class="btn btn--action" style="margin: 1em 0" @click="copyStockToClipboard()">
<i class="fa-regular fa-copy"></i> {{ i18n.t('trains.stock-copy') }} <i class="fa-regular fa-copy"></i> {{ i18n.t('trains.stock-copy') }}
</button> </button>
<button class="btn btn--action" @click="toggleNumberPropositions()"> <button class="btn btn--action" style="margin: 1em 0" @click="toggleNumberPropositions()">
<i class="fa-regular fa-lightbulb"></i> {{ i18n.t('trains.number-propositions') }} <i class="fa-regular fa-lightbulb"></i> {{ i18n.t('trains.number-propositions') }}
</button> </button>
</div> </div>
<!-- Proposed numbers container --> <!-- Proposed numbers container -->
<transition name="view-anim"> <transition name="view-anim" class="propositions-container">
<DriverPropositions :chosenTrain="chosenTrain" v-if="arePropositionsVisible" /> <div v-if="arePropositionsVisible">
<h3 style="margin-bottom: 0.5em">{{ i18n.t('trains.number-propositions-header') }}</h3>
<div class="categories-select">
<button
v-for="(category, i) in availableCategories"
class="btn btn--option btn--action"
@click="selectCategory(i)"
:class="{ checked: i == chosenCategoryIndex }"
>
{{ category }}
</button>
</div>
<div v-if="numberPropositions.length > 0" class="propositions-numbers">
<div v-if="chosenCategory">
<b>{{ chosenCategory }} </b> -
{{ i18n.t(`categories.${chosenCategory.slice(0, 2)}`) }}
({{ i18n.t(`categories.${chosenCategory.slice(2)}`) }})
</div>
<div v-if="chosenCategoryRules">
<span v-if="chosenCategoryRules[0]"
>{{ i18n.t('trains.number-propositions-third-number') }}
<b class="text--primary">{{ chosenCategoryRules[0] }}</b> &bull;
</span>
<span
>{{
i18n.t('trains.number-propositions-last-nums', {
count: chosenCategoryRules[1].length
})
}}
<b class="text--primary">{{ chosenCategoryRules[1] }}</b> -
<b class="text--primary">{{ chosenCategoryRules[2] }}</b></span
>
</div>
<div style="margin-top: 0.5em">
<b>{{ i18n.t('trains.number-propositions-title') }}&nbsp;</b>
<i>{{ numberPropositions.join(', ') }}</i>
</div>
</div>
<div class="no-propositions" v-else>{{ i18n.t('trains.number-propositions-empty') }}</div>
</div>
</transition> </transition>
<StockList :trainStockList="chosenTrain.stockList" :key="chosenTrain.id" :showPreviews="true" /> <StockList :trainStockList="chosenTrain.stockList" />
<TrainSchedule :train="chosenTrain" /> <TrainSchedule :train="chosenTrain" />
</div> </div>
</template> </template>
@@ -27,15 +72,25 @@
import { PropType, ref } from 'vue'; import { PropType, ref } from 'vue';
import { Train } from '../../typings/common'; import { Train } from '../../typings/common';
import { useI18n } from 'vue-i18n'; import { useI18n } from 'vue-i18n';
import { useApiStore } from '../../store/apiStore';
import StockList from '../Global/StockList.vue'; import StockList from '../Global/StockList.vue';
import TrainSchedule from '../TrainsView/TrainSchedule.vue'; import TrainSchedule from '../TrainsView/TrainSchedule.vue';
import TrainInfo from '../TrainsView/TrainInfo.vue'; import TrainInfo from '../TrainsView/TrainInfo.vue';
import DriverPropositions from './DriverPropositions.vue';
import rulesJSON from '../../data/trainNumberRules.json';
import { computed } from 'vue';
import { watch } from 'vue';
const apiStore = useApiStore();
const i18n = useI18n(); const i18n = useI18n();
const arePropositionsVisible = ref(false); const arePropositionsVisible = ref(false);
const chosenCategoryIndex = ref(0);
const numberPropositions = ref<string[]>([]);
const chosenCategoryRules = ref<any[]>([]);
const props = defineProps({ const props = defineProps({
chosenTrain: { chosenTrain: {
@@ -64,7 +119,153 @@ function copyStockToClipboard() {
function toggleNumberPropositions() { function toggleNumberPropositions() {
arePropositionsVisible.value = !arePropositionsVisible.value; arePropositionsVisible.value = !arePropositionsVisible.value;
if (arePropositionsVisible.value) generateNumberPropositions();
} }
function selectCategory(i: number) {
chosenCategoryIndex.value = i;
generateNumberPropositions();
}
function generateNumberPropositions() {
const categoryCode = chosenCategory.value?.slice(0, 2);
const trainNoStr = props.chosenTrain.trainNo.toString();
// Get category rules
const rules = categoryCode
? ((rulesJSON.categoriesRules as any)[categoryCode] as any[])
: undefined;
if (!categoryCode || !rules) {
numberPropositions.value.length = 0;
chosenCategoryRules.value.length = 0;
return;
}
const [thirdNumber, minRange, maxRange] = rules;
const propositionsArr: string[] = [];
for (let i = 0; i < 5; i++) {
let generatedNumStr = '';
generatedNumStr += trainNoStr.at(0) ?? Math.floor(Math.random() * 10);
generatedNumStr += trainNoStr.at(1) ?? Math.floor(Math.random() * 10);
// Third number
generatedNumStr += thirdNumber ?? '';
// Remaining numbers
const rangeNums = minRange?.length ?? 3;
const randRange = Math.floor(
Math.random() * (Number(maxRange) - Number(minRange)) + Number(minRange)
).toString();
const leadingZeros = new Array(Math.abs(randRange.toString().length - rangeNums))
.fill('0')
.join('');
generatedNumStr += `${leadingZeros}${randRange}`;
const isNumberTaken =
apiStore.activeData?.trains?.some((t) => t.trainNo.toString() == generatedNumStr) ?? false;
if (!isNumberTaken) {
propositionsArr.push(generatedNumStr);
} else {
i--;
}
if (Number(randRange) > Number(maxRange)) break;
}
numberPropositions.value = propositionsArr;
chosenCategoryRules.value = rules;
}
const chosenCategory = computed(() => {
return availableCategories.value.at(chosenCategoryIndex.value);
});
const availableCategories = computed(() => {
const stockList = props.chosenTrain.stockList;
const headVehicle = stockList.at(0)?.split('-')[0] ?? '';
let availableCategories: string[] = [];
let categoryTraction = 'E';
let vehicleTypesSet = new Set<string>();
let wagonsNamesSet = new Set<string>();
let cargoNamesSet = new Set<string>();
for (const stockName of stockList) {
const [vehicleName, ...cargoList] = stockName.split(':');
const vehicleData = apiStore.vehiclesData?.vehicles.find((v) => v.name == vehicleName);
if (!vehicleData) continue;
vehicleTypesSet.add(vehicleData.type);
if (vehicleData.type.startsWith('wagon-')) wagonsNamesSet.add(vehicleData.name.split('_')[0]);
if (cargoList !== undefined) cargoList.forEach((c) => cargoNamesSet.add(c.split('_')[0]));
}
let vehicleTypesArr = [...vehicleTypesSet];
let wagonsNamesArr = [...wagonsNamesSet];
// Traction
if (vehicleTypesArr[0] == 'loco-electric') categoryTraction = 'E';
else if (vehicleTypesArr[0] == 'loco-diesel') categoryTraction = 'S';
else if (vehicleTypesArr[0] == 'unit-electric') categoryTraction = 'J';
else categoryTraction = 'M';
// EMU / DMU - M*, R*, P*
if (vehicleTypesArr.length == 1 && (categoryTraction == 'J' || categoryTraction == 'M')) {
availableCategories.push('MO', 'MP', 'MM', 'RO', 'RP', 'RA', 'RM', 'PW');
}
// Only locos (up to 3) - LT, LP, LS
else if (stockList.length <= 3 && vehicleTypesArr.every((v) => v.startsWith('loco-'))) {
if (/^(EU|ET|201E|4E|SU|ST|M62|CTLR4C)/.test(headVehicle)) availableCategories.push('LT');
if (/^(EU|EP|SU|SP)/.test(headVehicle)) availableCategories.push('LP');
if (/^(SM)/.test(headVehicle)) availableCategories.push('LS');
}
// Only locos (more than 3) - TH
else if (stockList.length > 3 && vehicleTypesArr.every((v) => v.startsWith('loco-'))) {
availableCategories.push('TH');
}
// Loco(s) + passenger only wagons - M*, R*, E*, P*
else if (vehicleTypesArr.every((v) => v.startsWith('loco-') || v == 'wagon-passenger')) {
availableCategories.push('EI', 'EC', 'EN', 'MO', 'MP', 'MM', 'RO', 'RP', 'RA', 'RM', 'PW');
}
// Loco(s) + cargo only / mixed wagons - T*, Z*
else {
if (wagonsNamesArr.every((v) => /^(627Z|412Z)/.test(v)))
availableCategories.push('TC', 'TD', 'TS');
else if (stockList.slice(1).every((v) => /PKPE/.test(v))) {
availableCategories.push('ZU', 'ZN');
} else if (wagonsNamesArr.length < 3 || cargoNamesSet.size < 3) {
availableCategories.push('TM', 'TG', 'TS', 'TK');
} else {
availableCategories.push('TN', 'TR', 'TS', 'TK');
}
}
return availableCategories.map((c) => `${c}${categoryTraction}`);
});
watch(
computed(() => `${props.chosenTrain.trainNo}`),
() => {
chosenCategoryIndex.value = 0;
generateNumberPropositions();
}
);
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@@ -78,13 +279,49 @@ function toggleNumberPropositions() {
.train-stock-actions { .train-stock-actions {
display: flex; display: flex;
gap: 0.5em;
}
.propositions-container {
margin-bottom: 1em;
padding: 0.5em;
background-color: #111;
}
.categories-select {
display: inline-flex;
flex-wrap: wrap; flex-wrap: wrap;
gap: 0.5em; gap: 0.5em;
margin: 1em 0;
position: relative;
&::after {
content: '';
position: absolute;
bottom: calc(-0.5em);
left: 0;
width: 100%;
height: 2px;
background-color: #aaa;
}
}
.propositions-numbers {
margin-top: 1em;
}
.no-propositions {
margin-top: 1em;
color: #ccc;
} }
@include responsive.smallScreen { @include responsive.smallScreen {
.train-stock-actions { .propositions-container {
text-align: center;
}
.categories-select {
justify-content: center; justify-content: center;
} }
} }
+1 -5
View File
@@ -7,8 +7,6 @@
:vehicle-string="vehicleString" :vehicle-string="vehicleString"
:images="images" :images="images"
:image-fallbacks="imagesFallbacks" :image-fallbacks="imagesFallbacks"
:show-previews="showPreviews"
:thumbnail-size="thumbnailSize"
/> />
</li> </li>
</ul> </ul>
@@ -25,9 +23,7 @@ export default defineComponent({
props: { props: {
trainStockList: { type: Array as PropType<string[]>, required: true }, trainStockList: { type: Array as PropType<string[]>, required: true },
tractionOnly: { type: Boolean, required: false }, tractionOnly: { type: Boolean, required: false }
showPreviews: { type: Boolean },
thumbnailSize: { type: Number }
}, },
data() { data() {
+6 -11
View File
@@ -9,10 +9,9 @@
<img <img
v-for="(thumbnailImage, imageIndex) in images" v-for="(thumbnailImage, imageIndex) in images"
:src="`https://stacjownik.spythere.eu/static/thumbnails/${thumbnailImage}.png`" :src="`https://stacjownik.spythere.eu/static/thumbnails/${thumbnailImage}.png`"
:height="thumbnailSize || 70" height="70"
loading="lazy" loading="lazy"
:data-crosshair-cursor="showPreviews" data-tooltip-type="VehiclePreviewTooltip"
:data-tooltip-type="showPreviews ? 'VehiclePreviewTooltip' : ''"
:data-tooltip-content="vehicleString" :data-tooltip-content="vehicleString"
@error="onImageError($event, imageFallbacks[imageIndex])" @error="onImageError($event, imageFallbacks[imageIndex])"
@load="onImageLoad" @load="onImageLoad"
@@ -27,9 +26,7 @@ import { computed, PropType, Ref, ref } from 'vue';
const props = defineProps({ const props = defineProps({
vehicleString: { type: String, required: true }, vehicleString: { type: String, required: true },
images: { type: Object as PropType<string[]>, required: true }, images: { type: Object as PropType<string[]>, required: true },
imageFallbacks: { type: Object as PropType<string[]>, required: true }, imageFallbacks: { type: Object as PropType<string[]>, required: true }
showPreviews: { type: Boolean },
thumbnailSize: { type: Number }
}); });
const thumbRef = ref(null) as Ref<HTMLElement | null>; const thumbRef = ref(null) as Ref<HTMLElement | null>;
@@ -68,7 +65,7 @@ function onImageLoad() {
max-width: 90%; max-width: 90%;
text-align: center; text-align: center;
color: #aaa; color: #aaa;
font-size: 0.8em; font-size: 0.85em;
margin: 0 auto; margin: 0 auto;
padding: 0.25em 0; padding: 0.25em 0;
} }
@@ -77,10 +74,8 @@ function onImageLoad() {
display: flex; display: flex;
justify-content: center; justify-content: center;
align-items: flex-end; align-items: flex-end;
padding: 0.5em 0; cursor: crosshair;
&[data-crosshair-cursor='true'] { padding: 0.5em 0;
cursor: crosshair;
}
} }
</style> </style>
@@ -18,20 +18,7 @@
</b> </b>
<span <span
v-if="isCreator(entry.dispatcherName)" v-if="apiStore.donatorsData.includes(entry.dispatcherName)"
data-tooltip-type="CreatorTooltip"
:data-tooltip-content="$t('donations.creator-message')"
>
<router-link
class="text--creator"
:to="`/journal/dispatchers?search-dispatcher=${entry.dispatcherName}`"
>
{{ entry.dispatcherName }}
</router-link>
</span>
<span
v-else-if="apiStore.donatorsData.includes(entry.dispatcherName)"
data-tooltip-type="DonatorTooltip" data-tooltip-type="DonatorTooltip"
:data-tooltip-content="$t('donations.dispatcher-message')" :data-tooltip-content="$t('donations.dispatcher-message')"
> >
@@ -135,7 +122,6 @@ import styleMixin from '../../../mixins/styleMixin';
import { useApiStore } from '../../../store/apiStore'; import { useApiStore } from '../../../store/apiStore';
import StationStatusBadge from '../../Global/StationStatusBadge.vue'; import StationStatusBadge from '../../Global/StationStatusBadge.vue';
import FlagIcon from '../../Global/FlagIcon.vue'; import FlagIcon from '../../Global/FlagIcon.vue';
import { isCreator } from '../../../utils/userUtils';
export default defineComponent({ export default defineComponent({
props: { props: {
@@ -148,7 +134,7 @@ export default defineComponent({
emits: ['toggleShowExtraInfo'], emits: ['toggleShowExtraInfo'],
data() { data() {
return { regions, apiStore: useApiStore(), isCreator }; return { regions, apiStore: useApiStore() };
}, },
methods: { methods: {
+14 -28
View File
@@ -120,15 +120,15 @@
</div> </div>
</section> </section>
</div> </div>
</div>
<div class="options_actions"> <div class="options_actions">
<button class="btn--action" @click="onResetButtonClick"> <button class="btn--action" @click="onResetButtonClick">
{{ $t('options.reset-button') }} {{ $t('options.reset-button') }}
</button> </button>
<button class="btn--action" @click="onSearchButtonConfirm"> <button class="btn--action" @click="onSearchButtonConfirm">
{{ $t('options.search-button') }} {{ $t('options.search-button') }}
</button> </button>
</div>
</div> </div>
</div> </div>
</transition> </transition>
@@ -269,9 +269,9 @@ export default defineComponent({
this.searchTimeout = window.setTimeout(async () => { this.searchTimeout = window.setTimeout(async () => {
try { try {
const suggestions: string[] = await this.apiStore.client.get( const suggestions: string[] = await (
`api/get${type}Suggestions?name=${value}` await this.apiStore.client!.get(`api/get${type}Suggestions?name=${value}`)
); ).data;
this[`${type}Suggestions`] = suggestions; this[`${type}Suggestions`] = suggestions;
} catch (error) { } catch (error) {
@@ -330,23 +330,9 @@ export default defineComponent({
<style lang="scss" scoped> <style lang="scss" scoped>
@use '../../styles/dropdown'; @use '../../styles/dropdown';
@use '../../styles/dropdown-filters'; @use '../../styles/dropdown-filters';
@use '../../styles/responsive';
.dropdown_wrapper { .filters-options > .dropdown_wrapper {
display: grid; height: calc(100vh - 19em);
grid-template-rows: 1fr auto; min-height: 500px;
overflow: hidden;
max-height: calc(100% - 4.5em);
top: 3.5em;
padding: 1em 0;
}
.options_content {
overflow: auto;
padding: 0 1em;
}
.options_actions {
padding: 0 1em;
} }
</style> </style>
@@ -64,18 +64,10 @@ function navigateToProfile() {
<style lang="scss" scoped> <style lang="scss" scoped>
@use '../../styles/dropdown'; @use '../../styles/dropdown';
@use '../../styles/dropdown-filters'; @use '../../styles/dropdown-filters';
@use '../../styles/responsive';
.dropdown_wrapper { .dropdown_wrapper {
left: auto; left: auto;
right: 0; right: 0;
max-width: 700px; max-width: 700px;
top: 3.5em;
}
@include responsive.smallScreen {
.dropdown_wrapper {
top: 6.25em;
}
} }
</style> </style>
@@ -129,7 +129,6 @@
: stockHistory[currentHistoryIndex].stockString : stockHistory[currentHistoryIndex].stockString
).split(';') ).split(';')
" "
:showPreviews="true"
/> />
</div> </div>
</div> </div>
@@ -201,20 +200,22 @@ const driverRouteLocation = computed<RouteLocationRaw | null>(() => {
async function fetchTimetableDetails() { async function fetchTimetableDetails() {
try { try {
const responseData = await apiStore.client.get<API.TimetableHistory.Response>( const responseData = await apiStore.client!.get<API.TimetableHistory.Response>(
'api/getTimetables', 'api/getTimetables',
{ {
timetableId: props.timetableEntry.id, params: {
returnType: 'detailed' timetableId: props.timetableEntry.id,
returnType: 'detailed'
}
} }
); );
if (!responseData || responseData.length != 1) { if (!responseData || responseData.data.length != 1) {
timetableDetails.value = null; timetableDetails.value = null;
return; return;
} }
timetableDetails.value = responseData[0]; timetableDetails.value = responseData.data[0];
} catch (error) { } catch (error) {
// this.dataStatus = Status.Data.Error; // this.dataStatus = Status.Data.Error;
console.error(error); console.error(error);
@@ -59,17 +59,7 @@
</strong> </strong>
<router-link <router-link
v-if="isCreator(timetable.driverName)" v-if="apiStore.donatorsData.includes(timetable.driverName)"
class="text--creator"
data-tooltip-type="CreatorTooltip"
:data-tooltip-content="$t('donations.creator-message')"
:to="`/journal/timetables?search-driver=${timetable.driverName}`"
>
<strong>{{ timetable.driverName }}</strong>
</router-link>
<router-link
v-else-if="apiStore.donatorsData.includes(timetable.driverName)"
class="text--donator" class="text--donator"
data-tooltip-type="DonatorTooltip" data-tooltip-type="DonatorTooltip"
:data-tooltip-content="$t('donations.driver-message')" :data-tooltip-content="$t('donations.driver-message')"
@@ -125,7 +115,6 @@ import styleMixin from '../../../mixins/styleMixin';
import { useApiStore } from '../../../store/apiStore'; import { useApiStore } from '../../../store/apiStore';
import trainCategoryMixin from '../../../mixins/trainCategoryMixin'; import trainCategoryMixin from '../../../mixins/trainCategoryMixin';
import FlagIcon from '../../Global/FlagIcon.vue'; import FlagIcon from '../../Global/FlagIcon.vue';
import { isCreator } from '../../../utils/userUtils';
export default defineComponent({ export default defineComponent({
components: { FlagIcon }, components: { FlagIcon },
@@ -133,8 +122,7 @@ export default defineComponent({
data() { data() {
return { return {
apiStore: useApiStore(), apiStore: useApiStore()
isCreator
}; };
}, },
+1 -2
View File
@@ -15,8 +15,7 @@ export namespace Journal {
| 'search-issuedFrom' | 'search-issuedFrom'
| 'search-terminatingAt' | 'search-terminatingAt'
| 'search-via' | 'search-via'
| 'select-categoryCode' | 'select-categoryCode';
| 'search-headUnit';
export type TimetableSearchType = { export type TimetableSearchType = {
[key in TimetableSearchKey]: string; [key in TimetableSearchKey]: string;
@@ -25,7 +25,7 @@
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { PropType, ref, useTemplateRef } from 'vue'; import { computed, onMounted, PropType, ref, useTemplateRef } from 'vue';
import { Status } from '../../typings/common'; import { Status } from '../../typings/common';
import Loading from '../Global/Loading.vue'; import Loading from '../Global/Loading.vue';
import { Td2API } from '../../typings/api'; import { Td2API } from '../../typings/api';
@@ -36,6 +36,10 @@ const props = defineProps({
} }
}); });
onMounted(() => {
console.log(avatarImageRef.value);
});
const avatarImageRef = useTemplateRef('avatarImageRef'); const avatarImageRef = useTemplateRef('avatarImageRef');
const avatarLoadingStatus = ref<Status.Data>(Status.Data.Loading); const avatarLoadingStatus = ref<Status.Data>(Status.Data.Loading);
@@ -5,10 +5,7 @@
<ProfilePlayerAvatar :playerTD2Info="playerTD2Info" /> <ProfilePlayerAvatar :playerTD2Info="playerTD2Info" />
<div> <div>
<h2 <h2 class="player-name-header" :class="{ 'text--donator': isPlayerDonator }">
class="player-name-header"
:class="{ 'text--donator': isPlayerDonator, 'text--creator': isPlayerCreator }"
>
<a :href="`https://td2.info.pl/profile/?u=${route.query.playerId}`" target="_blank"> <a :href="`https://td2.info.pl/profile/?u=${route.query.playerId}`" target="_blank">
<img <img
v-if="isPlayerDonator" v-if="isPlayerDonator"
@@ -235,7 +232,6 @@ import { useApiStore } from '../../store/apiStore';
import StationStatusBadge from '../Global/StationStatusBadge.vue'; import StationStatusBadge from '../Global/StationStatusBadge.vue';
import ProfilePlayerAvatar from './ProfilePlayerAvatar.vue'; import ProfilePlayerAvatar from './ProfilePlayerAvatar.vue';
import { getRegionNameById } from '../../utils/regionUtils'; import { getRegionNameById } from '../../utils/regionUtils';
import { isCreator } from '../../utils/userUtils';
const { t } = useI18n(); const { t } = useI18n();
@@ -261,8 +257,6 @@ const isPlayerDonator = computed(() =>
props.playerName ? apiStore.donatorsData.includes(props.playerName) : false props.playerName ? apiStore.donatorsData.includes(props.playerName) : false
); );
const isPlayerCreator = computed(() => (props.playerName ? isCreator(props.playerName) : false));
const activeDispatches = computed(() => { const activeDispatches = computed(() => {
if (!props.playerName) return []; if (!props.playerName) return [];
if (!apiStore.activeData || !apiStore.activeData.activeSceneries) return []; if (!apiStore.activeData || !apiStore.activeData.activeSceneries) return [];
@@ -127,8 +127,9 @@ export default defineComponent({
this.station?.name || this.onlineScenery?.name this.station?.name || this.onlineScenery?.name
}&countFrom=${countFrom}&countLimit=${countLimit}`; }&countFrom=${countFrom}&countLimit=${countLimit}`;
const historyAPIData: API.DispatcherHistory.Response = const historyAPIData: API.DispatcherHistory.Response = await (
await this.apiStore.client.get(requestString); await this.apiStore.client!.get(requestString)
).data;
this.dataStatus = Status.Data.Loaded; this.dataStatus = Status.Data.Loaded;
return historyAPIData; return historyAPIData;
+28 -11
View File
@@ -1,15 +1,19 @@
<template> <template>
<section class="info-header"> <section class="info-header">
<button class="btn btn-return" :title="$t('scenery.return-btn')" @click="onReturnButtonClick"> <button
class="btn btn-return"
:title="$t('scenery.return-btn')"
@click="onReturnButtonClick"
>
<img src="/images/icon-back.svg" alt="return button" /> <img src="/images/icon-back.svg" alt="return button" />
</button> </button>
<div class="scenery-name"> <a class="scenery-name" :href="station?.generalInfo?.url" target="_blank">
<a v-if="station?.generalInfo" :href="station.generalInfo.url" target="_blank"> {{ stationName.replace(/_/g, ' ') }}
{{ stationName.replace(/_/g, ' ') }} </a>
</a>
<span v-else> {{ stationName.replace(/_/g, ' ') }}</span> <div class="scenery-abbrev" v-if="station?.generalInfo?.abbr">
{{ $t('scenery.abbrev') }} <b>{{ station.generalInfo.abbr }}</b>
</div> </div>
<div class="scenery-hash" v-if="onlineScenery?.hash">#{{ onlineScenery.hash }}</div> <div class="scenery-hash" v-if="onlineScenery?.hash">#{{ onlineScenery.hash }}</div>
@@ -24,6 +28,12 @@ import { useRoute, useRouter } from 'vue-router';
const route = useRoute(); const route = useRoute();
const router = useRouter(); const router = useRouter();
const prevPath = ref('/');
onMounted(() => {
prevPath.value = (route.meta['prevPath'] as string) ?? '/';
});
defineProps({ defineProps({
station: { station: {
type: Object as PropType<Station> type: Object as PropType<Station>
@@ -40,7 +50,7 @@ defineProps({
}); });
function onReturnButtonClick() { function onReturnButtonClick() {
router.push('/'); router.push(prevPath.value);
} }
</script> </script>
@@ -51,14 +61,15 @@ function onReturnButtonClick() {
.btn-return { .btn-return {
$bgColor: #2b2b2b; $bgColor: #2b2b2b;
background-color: $bgColor; background-color: $bgColor;
margin-bottom: 0.5em;
img {
width: 2em;
}
&:hover { &:hover {
background-color: color.adjust($color: $bgColor, $lightness: 15%); background-color: color.adjust($color: $bgColor, $lightness: 15%);
} }
img {
height: 2em;
}
} }
.scenery-name { .scenery-name {
@@ -70,7 +81,13 @@ function onReturnButtonClick() {
text-transform: uppercase; text-transform: uppercase;
} }
.scenery-abbrev {
font-size: 1.3em;
color: #aaa;
}
.scenery-hash { .scenery-hash {
margin-top: 0.5em;
color: #aaa; color: #aaa;
font-size: 1.2em; font-size: 1.2em;
} }
+30 -38
View File
@@ -1,32 +1,29 @@
<template> <template>
<div class="scenery-info"> <div class="scenery-info">
<section> <section>
<div class="info-station-data" v-if="apiStore.dataStatuses.sceneries == Status.Data.Loaded"> <SceneryInfoIcons :station="station" />
<SceneryInfoIcons :station="station" /> <SceneryInfoGeneral :station="station" />
<SceneryInfoGeneral :station="station" /> <SceneryInfoRoutes v-if="station" :station="station" />
<SceneryInfoRoutes v-if="station" :station="station" /> <SceneryInfoAuthors :station="station" />
<SceneryInfoAuthors :station="station" />
</div>
<div class="info-station-loading" v-else> <div style="margin: 1em 0; height: 2px; background-color: white"></div>
<Loading />
</div>
<div class="info-divider"></div>
<!-- info dispatcher -->
<SceneryInfoDispatcher :onlineScenery="onlineScenery" /> <SceneryInfoDispatcher :onlineScenery="onlineScenery" />
<div class="info-online-lists"> <div class="info-lists">
<!-- user list -->
<SceneryInfoUserList :onlineScenery="onlineScenery" :station="station" /> <SceneryInfoUserList :onlineScenery="onlineScenery" :station="station" />
<!-- spawn list -->
<SceneryInfoSpawnList :onlineScenery="onlineScenery" /> <SceneryInfoSpawnList :onlineScenery="onlineScenery" />
</div> </div>
</section> </section>
</div> </div>
</template> </template>
<script lang="ts" setup> <script lang="ts">
import { PropType } from 'vue'; import { PropType, defineComponent } from 'vue';
import { ActiveScenery, Station, Status } from '../../typings/common';
import SceneryInfoDispatcher from './SceneryInfo/SceneryInfoDispatcher.vue'; import SceneryInfoDispatcher from './SceneryInfo/SceneryInfoDispatcher.vue';
import SceneryInfoIcons from './SceneryInfo/SceneryInfoIcons.vue'; import SceneryInfoIcons from './SceneryInfo/SceneryInfoIcons.vue';
@@ -35,18 +32,27 @@ import SceneryInfoSpawnList from './SceneryInfo/SceneryInfoSpawnList.vue';
import SceneryInfoRoutes from './SceneryInfo/SceneryInfoRoutes.vue'; import SceneryInfoRoutes from './SceneryInfo/SceneryInfoRoutes.vue';
import SceneryInfoGeneral from './SceneryInfo/SceneryInfoGeneral.vue'; import SceneryInfoGeneral from './SceneryInfo/SceneryInfoGeneral.vue';
import SceneryInfoAuthors from './SceneryInfo/SceneryInfoAuthors.vue'; import SceneryInfoAuthors from './SceneryInfo/SceneryInfoAuthors.vue';
import { useApiStore } from '../../store/apiStore';
import Loading from '../Global/Loading.vue';
const apiStore = useApiStore(); import { ActiveScenery, Station } from '../../typings/common';
defineProps({ export default defineComponent({
station: { components: {
type: Object as PropType<Station> SceneryInfoDispatcher,
SceneryInfoGeneral,
SceneryInfoIcons,
SceneryInfoAuthors,
SceneryInfoUserList,
SceneryInfoSpawnList,
SceneryInfoRoutes
}, },
props: {
station: {
type: Object as PropType<Station>
},
onlineScenery: { onlineScenery: {
type: Object as PropType<ActiveScenery> type: Object as PropType<ActiveScenery>
}
} }
}); });
</script> </script>
@@ -54,15 +60,7 @@ defineProps({
<style lang="scss" scoped> <style lang="scss" scoped>
@use '../../styles/responsive'; @use '../../styles/responsive';
.info-station-loading { .info-lists {
display: flex;
align-items: center;
justify-content: center;
min-height: 300px;
}
.info-online-lists {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
justify-content: space-around; justify-content: space-around;
@@ -70,12 +68,6 @@ defineProps({
margin-top: 1em; margin-top: 1em;
} }
.info-divider {
margin: 1em 0;
height: 3px;
background-color: #5b5b5b;
}
.scenery-topic a { .scenery-topic a {
font-weight: bold; font-weight: bold;
} }
@@ -9,16 +9,9 @@
</span> </span>
<router-link class="dispatcher-name" :to="`/profile?playerId=${onlineScenery.dispatcherId}`"> <router-link class="dispatcher-name" :to="`/profile?playerId=${onlineScenery.dispatcherId}`">
<span
class="text--creator"
v-if="isCreator(onlineScenery.dispatcherName)"
:title="$t('donations.creator-message')"
>
{{ onlineScenery.dispatcherName }}
</span>
<span <span
class="text--donator" class="text--donator"
v-else-if="apiStore.donatorsData.includes(onlineScenery.dispatcherName)" v-if="apiStore.donatorsData.includes(onlineScenery.dispatcherName)"
:title="$t('donations.dispatcher-message')" :title="$t('donations.dispatcher-message')"
> >
{{ onlineScenery.dispatcherName }} {{ onlineScenery.dispatcherName }}
@@ -58,7 +51,6 @@ import StationStatusBadge from '../../Global/StationStatusBadge.vue';
import { ActiveScenery } from '../../../typings/common'; import { ActiveScenery } from '../../../typings/common';
import { useApiStore } from '../../../store/apiStore'; import { useApiStore } from '../../../store/apiStore';
import FlagIcon from '../../Global/FlagIcon.vue'; import FlagIcon from '../../Global/FlagIcon.vue';
import { isCreator } from '../../../utils/userUtils';
export default defineComponent({ export default defineComponent({
mixins: [styleMixin, dateMixin, routerMixin], mixins: [styleMixin, dateMixin, routerMixin],
@@ -66,8 +58,7 @@ export default defineComponent({
data() { data() {
return { return {
apiStore: useApiStore(), apiStore: useApiStore()
isCreator
}; };
}, },
@@ -5,69 +5,51 @@
</div> </div>
<div v-else> <div v-else>
<div> <span>
<span> <b>{{ $t('availability.title') }}:</b>
<a {{ $t(`availability.${station.generalInfo.availability}`) }}
v-if="station?.generalInfo"
:href="station.generalInfo.url"
class="forum-link"
target="_blank"
>
{{ $t('scenery.forum-topic') }}
</a>
</span>
<span> <span v-if="station.generalInfo.reqLevel > -1">
&bull; -
<b>{{ $t('scenery.abbrev') }}</b> {{ station.generalInfo.abbr }} {{
$t(
'scenery.req-level',
{ lvl: station.generalInfo.reqLevel },
station.generalInfo.reqLevel
)
}}
</span> </span>
</span>
<span> <span>
&bull; <b>{{ $t('availability.title') }}:</b> &bull; <b>{{ $t('controls.title') }}:</b>
{{ $t(`availability.${station.generalInfo.availability}`) }} {{ $t(`controls.${station.generalInfo.controlType}`) }}
</span>
<span v-if="station.generalInfo.reqLevel > -1"> <span>
- &bull; <b>{{ $t('signals.title') }}:</b>
{{ {{ $t(`signals.${station.generalInfo.signalType}`) }}
$t( </span>
'scenery.req-level',
{ lvl: station.generalInfo.reqLevel },
station.generalInfo.reqLevel
)
}}
</span>
</span>
<span> <span v-if="station.generalInfo.lines">
&bull; <b>{{ $t('controls.title') }}:</b> &bull; <b>{{ $t('scenery.lines-title') }}:</b> {{ station.generalInfo.lines }}
{{ $t(`controls.${station.generalInfo.controlType}`) }} </span>
</span>
<span> <span v-if="station.generalInfo.project">
&bull; <b>{{ $t('signals.title') }}:</b> &bull; <b>{{ $t('scenery.project-title') }}: </b>
{{ $t(`signals.${station.generalInfo.signalType}`) }} <a
</span> style="color: salmon; text-decoration: underline; font-weight: bold"
:href="station.generalInfo.projectUrl"
target="_blank"
>
{{ station.generalInfo.project }}
</a>
</span>
<span v-if="station.generalInfo.lines"> <span v-if="additionalTools.length != 0">
&bull; <b>{{ $t('scenery.lines-title') }}:</b> {{ station.generalInfo.lines }} &bull; <b>{{ $t('scenery.additional-tools-title') }}: </b>
</span> {{ additionalTools.join(', ') }}
</span>
<span v-if="station.generalInfo.project">
&bull; <b>{{ $t('scenery.project-title') }}: </b>
<a
style="color: salmon; text-decoration: underline; font-weight: bold"
:href="station.generalInfo.projectUrl"
target="_blank"
>
{{ station.generalInfo.project }}
</a>
</span>
<span v-if="additionalTools.length != 0">
&bull; <b>{{ $t('scenery.additional-tools-title') }}: </b>
{{ additionalTools.join(', ') }}
</span>
</div>
</div> </div>
</section> </section>
</template> </template>
@@ -102,14 +84,9 @@ export default defineComponent({
display: flex; display: flex;
justify-content: center; justify-content: center;
flex-wrap: wrap; flex-wrap: wrap;
}
.scenery-abbrev { div {
font-size: 1.05em; margin: 0 0.15em;
} }
a.forum-link {
text-decoration: underline;
font-weight: bold;
} }
</style> </style>
@@ -1,101 +1,102 @@
<template> <template>
<section class="info-icons-section"> <section class="info-icons">
<div class="icons-box"> <span v-if="!station || !station.generalInfo">
<span v-if="!station || !station.generalInfo">
<img
class="icon-info"
src="/images/icon-unknown.svg"
alt="icon-unknown"
:title="$t('sceneries.info.unknown')"
/>
</span>
<span
v-if="station?.generalInfo && station?.generalInfo.reqLevel >= 0"
class="scenery-icon icon-info level"
:style="calculateExpStyles(station?.generalInfo.reqLevel)"
>
{{ station?.generalInfo.reqLevel >= 2 ? station?.generalInfo.reqLevel : 'L' }}
</span>
<img <img
v-if="station?.generalInfo?.availability == 'nonPublic'"
class="icon-info" class="icon-info"
src="/images/icon-lock.svg" src="/images/icon-unknown.svg"
alt="Non-public scenery" alt="icon-unknown"
:title="$t('sceneries.info.non-public')" :title="$t('sceneries.info.unknown')"
/> />
</span>
<img <span
v-if="station?.generalInfo?.availability == 'unavailable'" v-if="station?.generalInfo && station?.generalInfo.reqLevel >= 0"
class="icon-info" class="scenery-icon icon-info level"
src="/images/icon-unavailable.svg" :style="calculateExpStyle(station?.generalInfo.reqLevel)"
alt="Unavailable scenery" >
:title="$t('sceneries.info.unavailable')" {{ station?.generalInfo.reqLevel >= 2 ? station?.generalInfo.reqLevel : 'L' }}
/> </span>
<img <img
v-if="station?.generalInfo?.availability == 'abandoned'" v-if="station?.generalInfo?.availability == 'nonPublic'"
class="icon-info" class="icon-info"
src="/images/icon-abandoned.svg" src="/images/icon-lock.svg"
alt="Abandoned scenery" alt="Non-public scenery"
:title="$t('sceneries.info.abandoned')" :title="$t('sceneries.info.non-public')"
/> />
<span <img
v-if="station?.generalInfo" v-if="station?.generalInfo?.availability == 'unavailable'"
class="scenery-icon icon-info" class="icon-info"
:class="station?.generalInfo.controlType.replace('+', '-')" src="/images/icon-unavailable.svg"
:title=" alt="Unavailable scenery"
$t('sceneries.info.control-type') + $t(`controls.${station?.generalInfo.controlType}`) :title="$t('sceneries.info.unavailable')"
" />
>
{{ $t(`controls.abbrevs.${station.generalInfo.controlType}`) }}
</span>
<img <img
v-if="station?.generalInfo?.signalType" v-if="station?.generalInfo?.availability == 'abandoned'"
class="icon-info" class="icon-info"
:src="`/images/icon-${station.generalInfo.signalType}.svg`" src="/images/icon-abandoned.svg"
:alt="station.generalInfo.signalType" alt="Abandoned scenery"
:title="$t('sceneries.info.signals-type') + $t(`signals.${station.generalInfo.signalType}`)" :title="$t('sceneries.info.abandoned')"
/> />
<img <span
v-if="station?.generalInfo?.lines" v-if="station?.generalInfo"
class="icon-info" class="scenery-icon icon-info"
src="/images/icon-real.svg" :class="station?.generalInfo.controlType.replace('+', '-')"
alt="real scenery" :title="
:title="`${$t('sceneries.info.real')} ${station.generalInfo.lines}`" $t('sceneries.info.control-type') + $t(`controls.${station?.generalInfo.controlType}`)
/> "
>
{{ $t(`controls.abbrevs.${station.generalInfo.controlType}`) }}
</span>
<img <img
v-if="station?.generalInfo?.SUP" v-if="station?.generalInfo?.signalType"
class="icon-info" class="icon-info"
src="/images/icon-SUP.svg" :src="`/images/icon-${station.generalInfo.signalType}.svg`"
alt="SUP (RASP-UZK)" :alt="station.generalInfo.signalType"
:title="$t('sceneries.info.SUP')" :title="$t('sceneries.info.signals-type') + $t(`signals.${station.generalInfo.signalType}`)"
/> />
<img <img
v-if="station?.generalInfo?.ASDEK" v-if="station?.generalInfo?.lines"
class="icon-info" class="icon-info"
src="/images/icon-ASDEK.svg" src="/images/icon-real.svg"
alt="dSAT ASDEK" alt="real scenery"
:title="$t('sceneries.info.ASDEK')" :title="`${$t('sceneries.info.real')} ${station.generalInfo.lines}`"
/> />
</div>
<img
v-if="station?.generalInfo?.SUP"
class="icon-info"
src="/images/icon-SUP.svg"
alt="SUP (RASP-UZK)"
:title="$t('sceneries.info.SUP')"
/>
<img
v-if="station?.generalInfo?.ASDEK"
class="icon-info"
src="/images/icon-ASDEK.svg"
alt="dSAT ASDEK"
:title="$t('sceneries.info.ASDEK')"
/>
</section> </section>
</template> </template>
<script lang="ts" setup> <script lang="ts">
import { PropType } from 'vue'; import { PropType, defineComponent } from 'vue';
import styleMixin from '../../../mixins/styleMixin';
import { Station } from '../../../typings/common'; import { Station } from '../../../typings/common';
import { calculateExpStyles } from '../../../composables/badge';
defineProps({ export default defineComponent({
station: { mixins: [styleMixin],
type: Object as PropType<Station> props: {
station: {
type: Object as PropType<Station>
}
} }
}); });
</script> </script>
@@ -103,12 +104,12 @@ defineProps({
<style lang="scss" scoped> <style lang="scss" scoped>
@use '../../../styles/icons'; @use '../../../styles/icons';
.icons-box { .info-icons {
display: flex; display: flex;
justify-content: center; justify-content: center;
flex-wrap: wrap; flex-wrap: wrap;
margin: 0.5em; margin: 1em;
} }
.icon-info { .icon-info {
@@ -1,18 +1,18 @@
<template> <template>
<section class="info-routes" v-if="station.generalInfo"> <section class="info-routes" v-if="station.generalInfo">
<div class="routes one-way" v-if="singleRoutesAvailable.length > 0"> <div class="routes one-way" v-if="oneWayRoutes.length > 0">
<button <button
class="routes-btn" class="routes-btn"
@click="toggleRoutesVisibility('single')" @click="toggleRoutesVisibility('single')"
data-tooltip-type="BaseTooltip" data-tooltip-type="BaseTooltip"
:data-tooltip-content="`${showInternalSingleRoutes ? $t('scenery.btn-show-internal-routes') : $t('scenery.btn-hide-internal-routes')}`" :data-tooltip-content="`${showInternalSingleRoutes ? $t('scenery.btn-hide-internal-routes') : $t('scenery.btn-show-internal-routes')}`"
> >
<b>{{ $t('scenery.one-way-routes') }}</b> <b>{{ $t('scenery.one-way-routes') }}</b>
<i class="fa-solid" :class="`${showInternalSingleRoutes ? 'fa-eye' : 'fa-eye-slash'}`"></i> <i class="fa-solid" :class="`${showInternalSingleRoutes ? 'fa-eye' : 'fa-eye-slash'}`"></i>
</button> </button>
<ul class="routes-list"> <ul class="routes-list">
<li v-for="route in singleRoutesFiltered" :key="route.routeName"> <li v-for="route in oneWayRoutes" :key="route.routeName">
<span :class="{ 'no-catenary': !route.isElectric, internal: route.isInternal }"> <span :class="{ 'no-catenary': !route.isElectric, internal: route.isInternal }">
{{ route.routeName }}</span {{ route.routeName }}</span
> >
@@ -24,29 +24,22 @@
</span> </span>
<span v-if="route.isRouteSBL" class="sbl">SBL</span> <span v-if="route.isRouteSBL" class="sbl">SBL</span>
</li> </li>
<li v-if="singleRoutesFiltered.length == 0">
<span class="routes-hidden">
<i class="fa-solid fa-eye-slash"></i>
{{ $t('scenery.routes-hidden') }}
</span>
</li>
</ul> </ul>
</div> </div>
<div class="routes two-way" v-if="doubleRoutesAvailable.length > 0"> <div class="routes two-way" v-if="twoWayRoutes.length > 0">
<button <button
class="routes-btn" class="routes-btn"
@click="toggleRoutesVisibility('double')" @click="toggleRoutesVisibility('double')"
data-tooltip-type="BaseTooltip" data-tooltip-type="BaseTooltip"
:data-tooltip-content="`${showInternalDoubleRoutes ? $t('scenery.btn-show-internal-routes') : $t('scenery.btn-hide-internal-routes')}`" :data-tooltip-content="`${showInternalDoubleRoutes ? $t('scenery.btn-hide-internal-routes') : $t('scenery.btn-show-internal-routes')}`"
> >
<b>{{ $t('scenery.two-way-routes') }}</b> <b>{{ $t('scenery.two-way-routes') }}</b>
<i class="fa-solid" :class="`${showInternalDoubleRoutes ? 'fa-eye' : 'fa-eye-slash'}`"></i> <i class="fa-solid" :class="`${showInternalDoubleRoutes ? 'fa-eye' : 'fa-eye-slash'}`"></i>
</button> </button>
<ul class="routes-list"> <ul class="routes-list">
<li v-for="route in doubleRoutesFiltered" :key="route.routeName"> <li v-for="route in twoWayRoutes" :key="route.routeName">
<span :class="{ 'no-catenary': !route.isElectric, internal: route.isInternal }"> <span :class="{ 'no-catenary': !route.isElectric, internal: route.isInternal }">
{{ route.routeName }} {{ route.routeName }}
</span> </span>
@@ -61,13 +54,6 @@
</span> </span>
<span v-if="route.isRouteSBL" class="sbl">SBL</span> <span v-if="route.isRouteSBL" class="sbl">SBL</span>
</li> </li>
<li v-if="doubleRoutesFiltered.length == 0">
<span class="routes-hidden">
<i class="fa-solid fa-eye-slash"></i>
{{ $t('scenery.routes-hidden') }}
</span>
</li>
</ul> </ul>
</div> </div>
</section> </section>
@@ -116,32 +102,20 @@ export default defineComponent({
}, },
computed: { computed: {
singleRoutesAvailable() { oneWayRoutes() {
return ( return (
this.station.generalInfo?.routes.single this.station.generalInfo?.routes.single
.filter((r) => !r.hidden) .filter((r) => !r.isInternal || r.isInternal == this.showInternalSingleRoutes)
.sort((r1, r2) => r1.routeName.localeCompare(r2.routeName)) ?? [] .sort((r1, r2) => r1.routeName.localeCompare(r2.routeName)) ?? []
); );
}, },
doubleRoutesAvailable() { twoWayRoutes() {
return ( return (
this.station.generalInfo?.routes.double this.station.generalInfo?.routes.double
.filter((r) => !r.hidden) .filter((r) => !r.isInternal || r.isInternal == this.showInternalDoubleRoutes)
.sort((r1, r2) => r1.routeName.localeCompare(r2.routeName)) ?? [] .sort((r1, r2) => r1.routeName.localeCompare(r2.routeName)) ?? []
); );
},
singleRoutesFiltered() {
return this.singleRoutesAvailable.filter(
(r) => this.showInternalSingleRoutes || !r.isInternal
);
},
doubleRoutesFiltered() {
return this.doubleRoutesAvailable.filter(
(r) => this.showInternalDoubleRoutes || !r.isInternal
);
} }
} }
}); });
@@ -180,6 +154,11 @@ ul.routes-list {
li { li {
margin: 0.5em 0.25em; margin: 0.5em 0.25em;
cursor: pointer;
user-select: none;
-moz-user-select: none;
-webkit-user-select: none;
& > span { & > span {
padding: 0.2em; padding: 0.2em;
@@ -203,16 +182,11 @@ ul.routes-list {
background-color: #303030; background-color: #303030;
color: #cfcfcf; color: #cfcfcf;
} }
&.sbl { &.sbl {
color: var(--clr-primary); color: var(--clr-primary);
background-color: #404040; background-color: #404040;
} }
&.routes-hidden {
background-color: #4b4b4b;
}
&:last-child { &:last-child {
border-radius: 0 0.5em 0.5em 0; border-radius: 0 0.5em 0.5em 0;
} }
@@ -29,8 +29,7 @@
<i <i
v-if=" v-if="
train.timetableData != undefined && train.timetableData != undefined &&
train.lastSeen <= Date.now() - 60000 && (train.lastSeen <= Date.now() - 60000 || !train.online)
!train.online
" "
class="fa-solid fa-user-slash" class="fa-solid fa-user-slash"
style="color: lightcoral" style="color: lightcoral"
+503 -23
View File
@@ -1,18 +1,247 @@
<template> <template>
<section class="scenery-timetable"> <section class="scenery-timetable">
<SceneryTimetableHeader <div class="timetable-header">
:station="station" <h3>
:onlineScenery="onlineScenery" <img src="/images/icon-timetable.svg" alt="icon-timetable" />
:chosenCheckpoint="chosenCheckpoint" <span>{{ $t('scenery.timetables') }}</span>
:showStockThumbnails="showStockThumbnails"
/>
<SceneryTimetableList <span>
:station="station" <span class="text--primary">{{ onlineScenery?.scheduledTrainCount.all ?? 0 }}</span>
:onlineScenery="onlineScenery" <span> / </span>
:chosenCheckpoint="chosenCheckpoint" <span class="text--grayed">
:showStockThumbnails="showStockThumbnails" {{ onlineScenery?.scheduledTrainCount.confirmed ?? 0 }}
/> </span>
</span>
<span class="header_links" v-if="station && onlineScenery">
<a
:href="generatorHref"
target="_blank"
data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t('scenery.gnr-link')}</b>`"
>
<img src="/images/icon-gnr.svg" alt="GeneraTOR app icon" />
</a>
<a
:href="pragotronHref"
target="_blank"
data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t('scenery.pragotron-link')}</b>`"
>
<img src="/images/icon-pragotron.svg" alt="icon-pragotron" />
</a>
<a
:href="tabliceZbiorczeHref"
target="_blank"
data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t('scenery.tablice-link')}</b>`"
>
<img src="/images/icon-tablice.ico" alt="icon-tablice" />
</a>
</span>
</h3>
<div class="timetable-checkpoints" v-if="station?.generalInfo?.checkpoints">
<template v-for="(ch, i) in station.generalInfo.checkpoints" :key="i">
<template v-if="i > 0">&bull;</template>
<router-link
class="checkpoint-item"
:class="{ current: chosenCheckpoint === ch }"
:to="`/scenery?station=${station.name}&checkpoint=${ch}`"
>{{ ch }}</router-link
>
</template>
</div>
<div class="timetable-checkpoints" v-else-if="onlineScenery">
<template v-for="(ch, i) in onlineScenery.missingCheckpoints" :key="i">
<template v-if="i > 0">&bull;</template>
<router-link
class="checkpoint-item"
:class="{ current: chosenCheckpoint === ch }"
:to="`/scenery?station=${onlineScenery.name}&checkpoint=${ch}`"
>{{ ch }}</router-link
>
</template>
</div>
</div>
<div class="timetable-list">
<transition-group name="list-anim">
<div
v-if="apiStore.dataStatuses.connection == 0 && sceneryTimetables.length == 0"
style="padding-bottom: 5em"
key="list-loading"
>
<Loading />
</div>
<span
class="timetable-item empty"
v-else-if="sceneryTimetables.length == 0 && !onlineScenery"
key="list-offline"
>
{{ $t('scenery.offline') }}
</span>
<div
class="timetable-item empty"
v-else-if="sceneryTimetables.length == 0"
key="list-no-timetables"
>
{{ $t('scenery.no-timetables') }}
</div>
<router-link
class="timetable-item"
v-else
v-for="(row, i) in sceneryTimetables"
:key="row.train.id + i"
tabindex="0"
:to="row.train.driverRouteLocation"
>
<span class="timetable-general">
<span class="general-info">
<div class="info-train">
<!-- Cargo warnings & details badges -->
<span
class="train-badge twr"
v-if="row.train.timetableData!.twr"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('warnings.TWR')"
>
TWR
</span>
<span
class="train-badge tn"
v-if="row.train.timetableData!.hasDangerousCargo"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('warnings.TN')"
>
TN
</span>
<span
class="train-badge pn"
v-if="row.train.timetableData!.hasExtraDeliveries"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('warnings.PN')"
>
PN
</span>
<!-- Train info -->
<span
data-tooltip-type="TrainInfoTooltip"
:data-tooltip-content="row.train.id"
class="tooltip-help"
>
<b class="text--primary">
{{ row.train.timetableData!.category }}
</b>
<b>&nbsp;{{ row.train.trainNo }}</b>
&bull;
{{ row.train.driverName }}
<i
class="fa-solid fa-user-slash"
style="color: salmon"
v-if="!row.train.online && row.train.lastSeen <= Date.now() - 60000"
></i>
</span>
<!-- Train stop comments -->
<span
v-if="row.checkpointStop.comments"
class="stop-comments-icon"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="row.checkpointStop.comments"
>
<img src="/images/icon-warning.svg" />
</span>
</div>
<div class="info-route">
<strong>{{ row.train.timetableData!.route.replace('|', ' - ') }}</strong>
</div>
<ScheduledTrainStatus :sceneryTimetableRow="row" />
</span>
</span>
<span class="timetable-schedule">
<span class="schedule-arrival">
<span class="arrival-time begins" v-if="row.checkpointStop.beginsHere">
{{ $t('timetables.begins') }}
</span>
<span class="arrival-time" v-else>
<div v-if="row.checkpointStop.arrivalDelay == 0">
<span>{{ timestampToString(row.checkpointStop.arrivalTimestamp) }}</span>
</div>
<div v-else>
<div>
<s style="margin-right: 0.2em" class="text--grayed">{{
timestampToString(row.checkpointStop.arrivalTimestamp)
}}</s>
</div>
<span>
{{ timestampToString(row.checkpointStop.arrivalRealTimestamp) }}
({{ row.checkpointStop.arrivalDelay > 0 ? '+' : ''
}}{{ row.checkpointStop.arrivalDelay }})
</span>
</div>
</span>
</span>
<span class="schedule-stop">
<span class="stop-connection">
{{ row.currentElement.arrivalRouteExt }}
</span>
<span class="stop-time">
{{ row.checkpointStop.stopTime || '' }}
{{ row.checkpointStop.stopTime ? row.checkpointStop.stopType || 'pt' : '' }}
</span>
<span class="stop-connection">
{{ row.currentElement.departureRouteExt }}
</span>
</span>
<span class="schedule-departure">
<span class="departure-time terminates" v-if="row.checkpointStop.terminatesHere">
{{ $t('timetables.terminates') }}
</span>
<span class="departure-time" v-else>
<div v-if="row.checkpointStop.departureDelay == 0">
<span>{{ timestampToString(row.checkpointStop.departureTimestamp) }}</span>
</div>
<div v-else>
<div>
<s style="margin-right: 0.2em" class="text--grayed">{{
timestampToString(row.checkpointStop.departureTimestamp)
}}</s>
</div>
<span>
{{ timestampToString(row.checkpointStop.departureRealTimestamp) }}
({{ row.checkpointStop.departureDelay > 0 ? '+' : ''
}}{{ row.checkpointStop.departureDelay }})
</span>
</div>
</span>
</span>
</span>
</router-link>
</transition-group>
</div>
</section> </section>
</template> </template>
@@ -20,21 +249,21 @@
import { computed, defineComponent, PropType, ref } from 'vue'; import { computed, defineComponent, PropType, ref } from 'vue';
import { useRoute } from 'vue-router'; import { useRoute } from 'vue-router';
import SceneryTimetableHeader from './SceneryTimetable/SceneryTimetableHeader.vue'; import Loading from '../Global/Loading.vue';
import dateMixin from '../../mixins/dateMixin'; import dateMixin from '../../mixins/dateMixin';
import routerMixin from '../../mixins/routerMixin'; import routerMixin from '../../mixins/routerMixin';
import trainCategoryMixin from '../../mixins/trainCategoryMixin'; import trainCategoryMixin from '../../mixins/trainCategoryMixin';
import { useMainStore } from '../../store/mainStore'; import { useMainStore } from '../../store/mainStore';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
import { ActiveScenery, Station } from '../../typings/common'; import ScheduledTrainStatus from './ScheduledTrainStatus.vue';
import SceneryTimetableList from './SceneryTimetable/SceneryTimetableList.vue'; import { SceneryTimetableRow } from './typings';
import StorageManager from '../../managers/storageManager'; import { ActiveScenery, Station, TooltipTrainInfo, Train } from '../../typings/common';
import { getTrainStopStatus, stopStatusPriority } from './utils';
export default defineComponent({ export default defineComponent({
name: 'SceneryTimetable', name: 'SceneryTimetable',
components: { SceneryTimetableHeader, SceneryTimetableList }, components: { Loading, ScheduledTrainStatus },
mixins: [dateMixin, routerMixin, trainCategoryMixin], mixins: [dateMixin, routerMixin, trainCategoryMixin],
@@ -48,8 +277,7 @@ export default defineComponent({
}, },
data: () => ({ data: () => ({
listOpen: false, listOpen: false
showStockThumbnails: false
}), }),
activated() { activated() {
@@ -85,6 +313,69 @@ export default defineComponent({
}; };
}, },
computed: {
tabliceZbiorczeHref() {
let url = `https://tablice-td2.web.app/?station=${this.station!.name}`;
if (this.chosenCheckpoint) url += `&checkpoint=${this.chosenCheckpoint}`;
return url;
},
pragotronHref() {
let url = `https://pragotron-td2.web.app/board?name=${this.station!.name}&region=${this.mainStore.region.id}`;
if (this.chosenCheckpoint) url += `&checkpoint=${this.chosenCheckpoint}`;
return url;
},
generatorHref() {
return `https://generator-td2.web.app/?sceneryId=${this.onlineScenery!.name}|${this.onlineScenery!.region}`;
},
sceneryTimetables(): SceneryTimetableRow[] {
if (!this.onlineScenery) return [];
const sceneryName = this.$route.query['station']?.toString().replace(/_/g, ' ') ?? '';
return this.onlineScenery.scheduledTrains
.filter(
(ct) =>
// ct.timetablePathElement.stationName == sceneryName &&
ct.train.region == this.mainStore.region.id &&
this.chosenCheckpoint &&
ct.checkpointStop.stopNameRAW.toLowerCase() == this.chosenCheckpoint.toLowerCase()
)
.map((ct) => {
const trainStopStatus = getTrainStopStatus(
ct.checkpointStop,
ct.train.currentStationName,
sceneryName
);
return {
checkpointStop: ct.checkpointStop,
train: ct.train,
prevElement: ct.previousSceneryElement,
nextElement: ct.nextSceneryElement,
currentElement: ct.timetablePathElement,
status: trainStopStatus
};
})
.sort((a, b) => {
if (stopStatusPriority.indexOf(a.status) - stopStatusPriority.indexOf(b.status) < 0)
return -1;
if (stopStatusPriority.indexOf(a.status) - stopStatusPriority.indexOf(b.status) > 0)
return 1;
if (a.checkpointStop.arrivalTimestamp > b.checkpointStop.arrivalTimestamp) return 1;
if (a.checkpointStop.arrivalTimestamp < b.checkpointStop.arrivalTimestamp) return -1;
return a.checkpointStop.departureTimestamp > b.checkpointStop.departureTimestamp ? 1 : -1;
});
}
},
methods: { methods: {
loadSelectedOption() { loadSelectedOption() {
const queryCheckpoint = this.$route.query['checkpoint']?.toString(); const queryCheckpoint = this.$route.query['checkpoint']?.toString();
@@ -111,16 +402,205 @@ export default defineComponent({
checkpointsListRef[0] ?? checkpointsListRef[0] ??
sceneryName; sceneryName;
} }
},
setCheckpoint(cp: string) {
this.chosenCheckpoint = cp;
} }
} }
}); });
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@use '../../styles/responsive';
@use '../../styles/animations';
@use '../../styles/badge';
.scenery-timetable { .scenery-timetable {
display: grid;
height: 100%; height: 100%;
overflow: hidden; overflow-y: scroll;
grid-template-rows: auto 1fr; padding: 0 0.5em;
}
.timetable-header {
position: sticky;
top: 0;
z-index: 99;
background-color: #181818;
padding: 0.5em;
img {
width: 25px;
vertical-align: middle;
}
h3 {
display: flex;
justify-content: center;
flex-wrap: wrap;
align-items: center;
gap: 0.5em;
font-size: 1.3em;
}
}
.header_links {
display: flex;
gap: 0.25em;
margin-left: 0.5em;
}
.timetable {
&-count {
margin-left: 0.5em;
}
&-item {
margin: 0.5em auto;
padding: 0.5em;
max-width: 1100px;
display: grid;
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
gap: 1.2em 0.5em;
overflow: hidden;
background: #353535;
z-index: 10;
&.empty {
padding: 1rem;
font-size: 1.2em;
color: #bbb;
}
}
&-general {
display: flex;
align-items: center;
justify-content: space-between;
text-align: left;
}
&-schedule {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 0.2em;
align-items: center;
width: 100%;
max-width: 400px;
margin: 0 auto;
}
}
.timetable-checkpoints {
display: flex;
justify-content: center;
gap: 0.5em;
flex-wrap: wrap;
font-size: 1.1em;
margin-top: 0.5em;
}
.checkpoint-item {
color: #aaa;
display: inline;
&:hover {
color: white;
}
&.current {
font-weight: bold;
color: var(--clr-primary);
}
}
.timetable-list {
position: relative;
}
.general-info {
display: flex;
flex-direction: column;
flex-wrap: wrap;
}
.info-train {
display: flex;
flex-wrap: wrap;
gap: 0.25em;
}
.info-train > .train-badge {
font-size: 0.85em;
}
.info-number {
color: var(--clr-primary);
}
.info-route {
width: 100%;
margin-top: 0.25em;
}
.stop-comments-icon > img {
width: 1.3em;
vertical-align: top;
}
.schedule {
&-arrival,
&-departure {
font-size: 1.15em;
}
&-stop {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 0.5em;
align-items: end;
.stop-connection {
font-size: 0.95em;
}
.stop-time {
position: relative;
inline-size: max-content;
align-self: center;
font-size: 0.9em;
color: var(--clr-primary);
&::after {
content: '\027F6';
display: block;
font-size: 2.2em;
line-height: 0.65em;
}
}
}
}
.arrival-time.begins,
.departure-time.terminates {
font-size: 0.85em;
}
@include responsive.smallScreen {
.timetable-item {
grid-template-columns: 1fr;
}
} }
</style> </style>
@@ -1,54 +0,0 @@
<template>
<div class="scenery-timetable-header">
<h3>
<img src="/images/icon-timetable.svg" alt="icon-timetable" />
<span>{{ $t('scenery.timetables') }}</span>
<span>
<span class="text--primary">{{ onlineScenery?.scheduledTrainCount.all ?? 0 }}</span>
<span> / </span>
<span class="text--grayed">
{{ onlineScenery?.scheduledTrainCount.confirmed ?? 0 }}
</span>
</span>
</h3>
</div>
</template>
<script lang="ts" setup>
import { computed, PropType } from 'vue';
import { Station, ActiveScenery } from '../../../typings/common';
import { useMainStore } from '../../../store/mainStore';
const props = defineProps({
station: {
type: Object as PropType<Station>
},
onlineScenery: {
type: Object as PropType<ActiveScenery>
},
chosenCheckpoint: {
type: String,
required: true
}
});
</script>
<style lang="scss" scoped>
.scenery-timetable-header {
background-color: #181818;
padding: 0.5em;
}
h3 {
display: flex;
justify-content: center;
flex-wrap: wrap;
align-items: center;
gap: 0.5em;
font-size: 1.3em;
}
</style>
@@ -1,567 +0,0 @@
<template>
<div class="scenery-timetable-list">
<!-- Checkpoints derived from station data -->
<div
class="timetable-checkpoints"
v-if="station?.generalInfo && station.generalInfo.checkpoints.length > 0"
>
<template v-for="(ch, i) in station.generalInfo.checkpoints" :key="i">
<template v-if="i > 0">&bull;</template>
<router-link
class="checkpoint-item"
:class="{ current: chosenCheckpoint === ch }"
:to="`/scenery?station=${station.name}&checkpoint=${ch}`"
>
{{ ch }}
</router-link>
</template>
</div>
<!-- Missing checkpoints if scenery is not in database -->
<div
class="timetable-checkpoints"
v-else-if="onlineScenery && onlineScenery.missingCheckpoints.length > 0"
>
<template v-for="(ch, i) in onlineScenery.missingCheckpoints" :key="i">
<template v-if="i > 0">&bull;</template>
<router-link
class="checkpoint-item"
:class="{ current: chosenCheckpoint === ch }"
:to="`/scenery?station=${onlineScenery.name}&checkpoint=${ch}`"
>
{{ ch }}
</router-link>
</template>
</div>
<div v-else></div>
<div class="list-container">
<transition-group name="list-anim">
<div
v-if="apiStore.dataStatuses.connection == 0 && sceneryTimetables.length == 0"
style="padding-bottom: 5em"
key="list-loading"
>
<Loading />
</div>
<div
class="timetable-item empty"
v-else-if="sceneryTimetables.length == 0 && !onlineScenery"
key="list-offline"
>
{{ $t('scenery.offline') }}
</div>
<div
class="timetable-item empty"
v-else-if="sceneryTimetables.length == 0"
key="list-no-timetables"
>
{{ $t('scenery.no-timetables') }}
</div>
<router-link
v-for="row in sceneryTimetables"
class="timetable-item"
:to="row.train.driverRouteLocation"
:key="row.train.id"
>
<div class="item-top">
<div class="top-general">
<span class="general-info">
<div class="info-train">
<!-- Cargo warnings & details badges -->
<span
class="train-badge twr"
v-if="row.train.timetableData!.twr"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('warnings.TWR')"
>
TWR
</span>
<span
class="train-badge tn"
v-if="row.train.timetableData!.hasDangerousCargo"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('warnings.TN')"
>
TN
</span>
<span
class="train-badge pn"
v-if="row.train.timetableData!.hasExtraDeliveries"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('warnings.PN')"
>
PN
</span>
<!-- Train info -->
<span
data-tooltip-type="TrainInfoTooltip"
:data-tooltip-content="row.train.id"
class="tooltip-help"
>
<b class="text--primary">
{{ row.train.timetableData!.category }}
</b>
<b>&nbsp;{{ row.train.trainNo }}</b>
&bull;
{{ row.train.driverName }}
<i
class="fa-solid fa-user-slash"
style="color: salmon"
v-if="!row.train.online && row.train.lastSeen <= Date.now() - 60000"
></i>
</span>
<!-- Train stop comments -->
<span
v-if="row.checkpointStop.comments"
class="stop-comments-icon"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="row.checkpointStop.comments"
>
<img src="/images/icon-warning.svg" />
</span>
</div>
<div class="info-route">
<strong>{{ row.train.timetableData!.route.replace('|', ' - ') }}</strong>
</div>
<ScheduledTrainStatus :sceneryTimetableRow="row" />
</span>
</div>
<div class="top-schedule">
<span class="schedule-arrival">
<span class="arrival-time begins" v-if="row.checkpointStop.beginsHere">
{{ $t('timetables.begins') }}
</span>
<span class="arrival-time" v-else>
<div v-if="row.checkpointStop.arrivalDelay == 0">
<span>{{ timestampToTimeString(row.checkpointStop.arrivalTimestamp) }}</span>
</div>
<div v-else>
<div>
<s style="margin-right: 0.2em" class="text--grayed">{{
timestampToTimeString(row.checkpointStop.arrivalTimestamp)
}}</s>
</div>
<span>
{{ timestampToTimeString(row.checkpointStop.arrivalRealTimestamp) }}
({{ row.checkpointStop.arrivalDelay > 0 ? '+' : ''
}}{{ row.checkpointStop.arrivalDelay }})
</span>
</div>
</span>
</span>
<span class="schedule-stop">
<span class="stop-connection">
{{ row.currentElement.arrivalRouteExt }}
</span>
<span class="stop-time">
{{ row.checkpointStop.stopTime || '' }}
{{ row.checkpointStop.stopTime ? row.checkpointStop.stopType || 'pt' : '' }}
</span>
<span class="stop-connection">
{{ row.currentElement.departureRouteExt }}
</span>
</span>
<span class="schedule-departure">
<span class="departure-time terminates" v-if="row.checkpointStop.terminatesHere">
{{ $t('timetables.terminates') }}
</span>
<span class="departure-time" v-else>
<div v-if="row.checkpointStop.departureDelay == 0">
<span>{{ timestampToTimeString(row.checkpointStop.departureTimestamp) }}</span>
</div>
<div v-else>
<div>
<s style="margin-right: 0.2em" class="text--grayed">{{
timestampToTimeString(row.checkpointStop.departureTimestamp)
}}</s>
</div>
<span>
{{ timestampToTimeString(row.checkpointStop.departureRealTimestamp) }}
({{ row.checkpointStop.departureDelay > 0 ? '+' : ''
}}{{ row.checkpointStop.departureDelay }})
</span>
</div>
</span>
</span>
</div>
</div>
<div class="item-stock-list" v-if="showStockThumbnails">
<StockList :trainStockList="row.train.stockList" :thumbnailSize="45" />
</div>
</router-link>
</transition-group>
</div>
<div class="list-actions" v-if="station && onlineScenery">
<a
:href="generatorHref"
target="_blank"
data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t('scenery.gnr-link')}</b>`"
>
<img src="/images/icon-gnr.svg" alt="GeneraTOR app icon" />
</a>
<a
:href="pragotronHref"
target="_blank"
data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t('scenery.pragotron-link')}</b>`"
>
<img src="/images/icon-pragotron.svg" alt="icon-pragotron" />
</a>
<a
:href="tabliceZbiorczeHref"
target="_blank"
data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t('scenery.tablice-link')}</b>`"
>
<img src="/images/icon-tablice.ico" alt="icon-tablice" />
</a>
<div class="list-divider"></div>
<button
class="thumbnails-btn"
data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t(`scenery.btn-${showStockThumbnails ? 'show' : 'hide'}-timetable-thumbnails`)}</b>`"
@click="toggleThumbnails"
>
<i class="fa-solid" :class="`${showStockThumbnails ? 'fa-expand' : 'fa-compress'}`"></i>
</button>
</div>
</div>
</template>
<script lang="ts" setup>
import { computed, ComputedRef, onMounted, PropType, ref } from 'vue';
import { Station, ActiveScenery } from '../../../typings/common';
import { SceneryTimetableRow } from '../typings';
import { getTrainStopStatus, stopStatusPriorities } from '../utils';
import { useRoute } from 'vue-router';
import { useMainStore } from '../../../store/mainStore';
import { useApiStore } from '../../../store/apiStore';
import { timestampToTimeString } from '../../../composables/time';
import ScheduledTrainStatus from './ScheduledTrainStatus.vue';
import Loading from '../../Global/Loading.vue';
import StockList from '../../Global/StockList.vue';
import StorageManager from '../../../managers/storageManager';
const props = defineProps({
station: {
type: Object as PropType<Station>
},
onlineScenery: {
type: Object as PropType<ActiveScenery>
},
chosenCheckpoint: {
type: String,
required: true
}
});
const route = useRoute();
const mainStore = useMainStore();
const apiStore = useApiStore();
const showStockThumbnails = ref(false);
onMounted(() => {
handleStockThumbnails();
});
const sceneryTimetables: ComputedRef<SceneryTimetableRow[]> = computed(() => {
if (!props.onlineScenery) return [];
const sceneryName = route.query['station']?.toString().replace(/_/g, ' ') ?? '';
return props.onlineScenery.scheduledTrains
.filter(
(ct) =>
// ct.timetablePathElement.stationName == sceneryName &&
ct.train.region == mainStore.region.id &&
props.chosenCheckpoint &&
ct.checkpointStop.stopNameRAW.toLowerCase() == props.chosenCheckpoint.toLowerCase()
)
.map((ct) => {
const trainStopStatus = getTrainStopStatus(
ct.checkpointStop,
ct.train.currentStationName,
sceneryName
);
return {
checkpointStop: ct.checkpointStop,
train: ct.train,
prevElement: ct.previousSceneryElement,
nextElement: ct.nextSceneryElement,
currentElement: ct.timetablePathElement,
status: trainStopStatus
};
})
.sort((a, b) => {
if (stopStatusPriorities.indexOf(a.status) - stopStatusPriorities.indexOf(b.status) < 0)
return -1;
if (stopStatusPriorities.indexOf(a.status) - stopStatusPriorities.indexOf(b.status) > 0)
return 1;
if (a.checkpointStop.arrivalTimestamp > b.checkpointStop.arrivalTimestamp) return 1;
if (a.checkpointStop.arrivalTimestamp < b.checkpointStop.arrivalTimestamp) return -1;
return a.checkpointStop.departureTimestamp > b.checkpointStop.departureTimestamp ? 1 : -1;
});
});
const tabliceZbiorczeHref = computed(() => {
let url = `https://tablice-td2.web.app/?station=${props.station!.name}`;
if (props.chosenCheckpoint) url += `&checkpoint=${props.chosenCheckpoint}`;
return url;
});
const pragotronHref = computed(() => {
let url = `https://pragotron-td2.spythere.eu/board?name=${props.station!.name}&region=${mainStore.region.id}`;
if (props.chosenCheckpoint) url += `&checkpoint=${props.chosenCheckpoint}`;
return url;
});
const generatorHref = computed(() => {
return `https://generator-td2.spythere.eu/?sceneryId=${props.onlineScenery!.name}|${props.onlineScenery!.region}`;
});
function handleStockThumbnails() {
const storageVal = StorageManager.getBooleanValue('showStockThumbnails');
showStockThumbnails.value = storageVal;
}
function toggleThumbnails() {
showStockThumbnails.value = !showStockThumbnails.value;
StorageManager.setBooleanValue('showStockThumbnails', showStockThumbnails.value);
}
</script>
<style lang="scss" scoped>
@use '../../../styles/responsive';
@use '../../../styles/animations';
@use '../../../styles/badge';
.scenery-timetable-list {
display: grid;
grid-template-rows: auto 1fr 40px;
overflow: hidden;
}
.top-general {
display: flex;
align-items: center;
justify-content: space-between;
text-align: left;
}
.top-schedule {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 0.2em;
align-items: center;
width: 100%;
max-width: 400px;
margin: 0 auto;
}
.timetable-checkpoints {
display: flex;
justify-content: center;
gap: 0.5em;
flex-wrap: wrap;
font-size: 1.1em;
margin: 0.5em 0;
}
.checkpoint-item {
color: #aaa;
display: inline;
&:hover {
color: white;
}
&.current {
font-weight: bold;
color: var(--clr-primary);
}
}
.list-container {
position: relative;
overflow-y: auto;
overflow-x: hidden;
margin-top: 0.5em;
padding: 2px;
width: 100%;
}
.timetable-item {
display: block;
margin-bottom: 0.5em;
padding: 0.35em;
width: 100%;
overflow: hidden;
background: #353535;
&.empty {
padding: 1rem;
font-size: 1.2em;
color: #bbb;
}
}
.timetable-item > .item-top {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(300px, 1fr));
gap: 1.2em 0.5em;
}
.timetable-item > .item-stock-list {
margin-top: 1em;
}
.general-info {
display: flex;
flex-direction: column;
flex-wrap: wrap;
}
.info-train {
display: flex;
flex-wrap: wrap;
gap: 0.25em;
}
.info-train > .train-badge {
font-size: 0.85em;
}
.info-number {
color: var(--clr-primary);
}
.info-route {
width: 100%;
margin-top: 0.25em;
}
.stop-comments-icon > img {
width: 1.3em;
vertical-align: top;
}
.schedule-arrival,
.schedule-departure {
font-size: 1.15em;
}
.schedule-stop {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 0.5em;
align-items: end;
.stop-connection {
font-size: 0.95em;
}
.stop-time {
position: relative;
inline-size: max-content;
align-self: center;
font-size: 0.9em;
color: var(--clr-primary);
&::after {
content: '\027F6';
display: block;
font-size: 2.2em;
line-height: 0.65em;
}
}
}
.arrival-time.begins,
.departure-time.terminates {
font-size: 0.85em;
}
.list-actions {
display: flex;
align-items: center;
gap: 0.5em;
margin-top: 0.5em;
.list-divider {
height: 80%;
width: 3px;
background-color: #6b6b6b;
}
img {
width: 25px;
height: 25px;
vertical-align: middle;
}
.thumbnails-btn {
width: 25px;
height: 25px;
font-size: 25px;
}
}
@include responsive.smallScreen {
.timetable-item {
grid-template-columns: 1fr;
}
.list-actions {
justify-content: center;
}
}
</style>
@@ -149,12 +149,11 @@ export default defineComponent({
requestFilters['returnType'] = 'short'; requestFilters['returnType'] = 'short';
try { try {
const response: API.TimetableHistory.ResponseShort = await this.apiStore.client.get( const response: API.TimetableHistory.ResponseShort = await (
'api/getTimetables', await this.apiStore.client!.get('api/getTimetables', {
requestFilters params: requestFilters
); })
).data;
console.log(response);
this.historyList = response; this.historyList = response;
@@ -1,213 +0,0 @@
<template>
<div class="scenery-top-list">
<h2 class="header">{{ t('scenery.top-list.header') }}</h2>
<div class="top-actions">
<div class="actions-modes">
<button
v-for="mode in availableModes"
:class="`btn btn--option ${mode == currentListMode ? 'checked' : ''}`"
@click="selectListMode(mode)"
>
{{ t(`scenery.top-list.mode-${mode}`) }}
</button>
</div>
<div class="actions-scopes">
<button
v-for="scope in availableScopes"
:class="`btn btn--option ${scope == currentListScope ? 'checked' : ''}`"
@click="selectListScope(scope)"
>
{{ t(`scenery.top-list.scope-${scope}`) }}
</button>
</div>
</div>
<div class="rating-list-wrapper">
<Loading v-if="listState == Status.Data.Loading" />
<div v-else-if="listState == Status.Data.Error">Ups, coś poszło nie tak...</div>
<ul v-else-if="bestScoreList.length > 0">
<li v-for="(value, i) in bestScoreList">
<div>
{{ t('ordinal', { count: i + 1 }) }} {{ t('scenery.top-list.place') }} -
<router-link :to="`/profile?playerId=${value.dispatcherId}`">{{
value.dispatcherName
}}</router-link>
</div>
<div>
<b class="text--primary" v-if="currentListMode == 'dutyCount'">{{
t('scenery.top-list.duty-count', value.value)
}}</b>
<b class="text--primary" v-else-if="currentListMode == 'dispatcherRating'">{{
t('scenery.top-list.dispatcher-rating', value.value)
}}</b>
<b class="text--primary" v-else>
{{ t('scenery.top-list.duration') }}
{{ humanizeDuration(value.value) }}
</b>
</div>
</li>
</ul>
<div v-else class="no-data">
<span v-if="currentListScope == 'name'">{{ t('scenery.top-list.no-data-general') }}</span>
<span v-else>{{ t('scenery.top-list.no-data-current-hash') }}</span>
</div>
</div>
</div>
</template>
<script lang="ts" setup>
import { onActivated, PropType, ref } from 'vue';
import { useI18n } from 'vue-i18n';
import { useApiStore } from '../../store/apiStore';
import { Station, ActiveScenery, Status } from '../../typings/common';
import Loading from '../Global/Loading.vue';
import { humanizeDuration } from '../../composables/time';
interface SceneryBestScoreItem {
dispatcherName: string;
dispatcherId: number;
value: number;
}
const { t } = useI18n();
const apiStore = useApiStore();
defineOptions({
name: 'SceneryTopList'
});
const props = defineProps({
station: {
type: Object as PropType<Station>
},
onlineScenery: {
type: Object as PropType<ActiveScenery>
}
});
const availableModes = ['dutyCount', 'dispatcherRating', 'dutyDuration'] as const;
const availableScopes = ['name', 'hash'] as const;
type ListMode = (typeof availableModes)[number];
type ListScope = (typeof availableScopes)[number];
const currentListMode = ref<ListMode>('dutyCount');
const currentListScope = ref<ListScope>('name');
const listState = ref<Status.Data>(Status.Data.Loading);
const bestScoreList = ref<SceneryBestScoreItem[]>([]);
onActivated(() => {
fetchTopDispatchersList();
});
function selectListMode(mode: ListMode) {
currentListMode.value = mode;
fetchTopDispatchersList();
}
function selectListScope(scope: ListScope) {
currentListScope.value = scope;
fetchTopDispatchersList();
}
async function fetchTopDispatchersList() {
const searchedStationValue =
currentListScope.value == 'name'
? props.station?.name
: apiStore.sceneryData.find((sc) => sc.name == props.station!.name)?.hash;
bestScoreList.value = [];
if (!searchedStationValue) {
listState.value = Status.Data.Loaded;
return;
}
try {
listState.value = Status.Data.Loading;
const response: SceneryBestScoreItem[] = await apiStore.client.get(`api/getSceneryBestScores`, {
[currentListScope.value]: searchedStationValue,
type: currentListMode.value,
countLimit: 40
});
bestScoreList.value = response;
listState.value = Status.Data.Loaded;
} catch (error) {
listState.value = Status.Data.Error;
console.error(error);
}
}
</script>
<style lang="scss" scoped>
.scenery-top-list {
display: grid;
grid-template-rows: auto auto 1fr;
overflow: hidden;
gap: 1em;
}
.top-actions {
display: flex;
justify-content: center;
flex-wrap: wrap;
gap: 0.5em 1.5em;
}
.actions-modes,
.actions-scopes {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
font-weight: bold;
}
.rating-list-wrapper {
overflow: auto;
}
.rating-list-wrapper > ul {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(350px, 1fr));
align-items: center;
gap: 0.65em;
padding-right: 0.5em;
}
.rating-list-wrapper > ul > li {
display: flex;
justify-content: center;
align-items: center;
flex-direction: column;
padding: 0.25em;
background-color: #2b2b2b;
height: 100%;
line-height: 1.5em;
a {
font-weight: bold;
}
}
.no-data {
padding: 1em 0.5em;
font-size: 1.1em;
background-color: #333;
color: #ccc;
}
</style>
@@ -18,8 +18,8 @@
<script lang="ts"> <script lang="ts">
import { defineComponent, PropType } from 'vue'; import { defineComponent, PropType } from 'vue';
import { StopStatus } from '../../../typings/common'; import { StopStatus } from '../../typings/common';
import { SceneryTimetableRow } from '../typings'; import { SceneryTimetableRow } from './typings';
export default defineComponent({ export default defineComponent({
props: { props: {
+6 -14
View File
@@ -1,6 +1,6 @@
import { StopStatus, TrainStop } from '../../typings/common'; import { StopStatus, TrainStop } from '../../typings/common';
export const stopStatusPriorities = [ export const stopStatusPriority = [
StopStatus.ONLINE, StopStatus.ONLINE,
StopStatus.STOPPED, StopStatus.STOPPED,
StopStatus.DEPARTED, StopStatus.DEPARTED,
@@ -18,31 +18,23 @@ export function getTrainStopStatus(
return StopStatus.TERMINATED; return StopStatus.TERMINATED;
} }
if ( if (!stopInfo.terminatesHere && stopInfo.confirmed && currentStationName == sceneryName) {
!stopInfo.terminatesHere &&
stopInfo.confirmed &&
currentStationName.startsWith(sceneryName)
) {
return StopStatus.DEPARTED; return StopStatus.DEPARTED;
} }
if ( if (!stopInfo.terminatesHere && stopInfo.confirmed && currentStationName != sceneryName) {
!stopInfo.terminatesHere &&
stopInfo.confirmed &&
!currentStationName.startsWith(sceneryName)
) {
return StopStatus.DEPARTED_AWAY; return StopStatus.DEPARTED_AWAY;
} }
if (currentStationName.startsWith(sceneryName) && !stopInfo.stopped) { if (currentStationName == sceneryName && !stopInfo.stopped) {
return StopStatus.ONLINE; return StopStatus.ONLINE;
} }
if (currentStationName.startsWith(sceneryName) && stopInfo.stopped) { if (currentStationName == sceneryName && stopInfo.stopped) {
return StopStatus.STOPPED; return StopStatus.STOPPED;
} }
if (!currentStationName.startsWith(sceneryName)) { if (currentStationName != sceneryName) {
return StopStatus.ARRIVING; return StopStatus.ARRIVING;
} }
@@ -1,154 +0,0 @@
<template>
<div class="filter-slider-container">
<input
class="slider"
v-for="slider in sliderGroupsOptions[sliderGroup]"
type="range"
:name="slider.id"
:id="slider.id"
:min="slider.minRange"
:max="slider.maxRange"
:step="slider.step"
v-model="filters[slider.id]"
/>
<div class="slider-track" @click="moveCloserSliderToMousePos"></div>
</div>
</template>
<script lang="ts" setup>
import { inject, PropType } from 'vue';
import { SliderGroup, sliderGroupsOptions } from '../../managers/stationFilterManager';
const filters = inject('StationsView_filters') as Record<string, any>;
const props = defineProps({
sliderGroup: {
type: String as PropType<SliderGroup>,
required: true
}
});
// Change slider value that's the closest one to the mouse position on the slider track click
function moveCloserSliderToMousePos(e: MouseEvent) {
const { clientX, target } = e;
const { minRange, maxRange, step } = sliderGroupsOptions[props.sliderGroup][0];
const boundingRect = (target as HTMLElement).getBoundingClientRect();
const mouseX = clientX - boundingRect.left;
const leftSliderValue = filters[sliderGroupsOptions[props.sliderGroup][0].id];
const rightSliderValue = filters[sliderGroupsOptions[props.sliderGroup][1].id];
let mouseValue = Math.round((maxRange - minRange) * (mouseX / boundingRect.width));
// Adjust mouse value to the closest step point (divide by 10, get rounded number, then multiply by step)
mouseValue = Math.round(mouseValue / step) * step;
let sliderIndex =
Math.abs(leftSliderValue - mouseValue) < Math.abs(rightSliderValue - mouseValue) ? 0 : 1;
filters[sliderGroupsOptions[props.sliderGroup][sliderIndex].id] = mouseValue;
}
</script>
<style lang="scss" scoped>
@use '../../styles/responsive';
.filter-slider-container {
position: relative;
padding: 0.5em;
height: 1.25em;
}
.slider-track {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
border-radius: 1em;
z-index: 10;
cursor: pointer;
background-color: #444;
transition: background-color 0.2s;
&:hover {
background-color: #4d4d4d;
}
}
.slider {
width: 100%;
height: 1.25em;
background: none;
outline: none;
border-radius: 1em;
padding: 0;
position: absolute;
left: 0;
top: 50%;
transform: translateY(-50%);
z-index: 100;
pointer-events: none;
cursor: pointer;
-webkit-appearance: none;
appearance: none;
&:hover ~ .slider-track {
background-color: #4d4d4d;
}
&:focus-visible {
outline: 1px solid white;
}
&::-webkit-slider-thumb {
-webkit-appearance: none;
appearance: none;
position: relative;
z-index: 100;
width: 1.25em;
height: 1.25em;
border-radius: 1em;
background: var(--clr-primary);
pointer-events: all;
}
&::-moz-range-thumb {
width: 1.25em;
height: 1.25em;
border-radius: 1em;
background: var(--clr-primary);
pointer-events: all;
}
// &:first-child::-webkit-slider-runnable-track {
// }
&::-moz-range-track {
position: relative;
z-index: -1;
width: 100%;
height: 5px;
cursor: pointer;
background: none;
border-radius: 1em;
}
// &:first-child::-moz-range-track {
// background: var(--clr-primary);
// }
}
</style>
+116 -39
View File
@@ -137,16 +137,20 @@
</section> </section>
<section class="card_sliders"> <section class="card_sliders">
<div class="option-slider" v-for="(sliderGroup, i) in sliderGroups" :key="i"> <div class="slider" v-for="(slider, i) in sliderStates" :key="i">
<FilterSlider :sliderGroup="sliderGroup" /> <input
class="slider-input"
<span class="slider-value"> type="range"
{{ filters[sliderGroupsOptions[sliderGroup][0].id] }} - :name="slider.id"
{{ filters[sliderGroupsOptions[sliderGroup][1].id] }} :id="slider.id"
</span> :min="slider.minRange"
:max="slider.maxRange"
:step="slider.step"
v-model.number="filters[slider.id]"
/>
<span class="slider-value">{{ filters[slider.id] }}</span>
<div class="slider-content"> <div class="slider-content">
{{ $t(`filters.sliders.${sliderGroups[i]}`) }} {{ $t(`filters.sliders.${slider.id}`) }}
</div> </div>
</div> </div>
</section> </section>
@@ -186,15 +190,13 @@ import routerMixin from '../../mixins/routerMixin';
import { useMainStore } from '../../store/mainStore'; import { useMainStore } from '../../store/mainStore';
import FilterOption from './FilterOption.vue'; import FilterOption from './FilterOption.vue';
import FilterSlider from './FilterSlider.vue';
import StorageManager from '../../managers/storageManager'; import StorageManager from '../../managers/storageManager';
import { import {
filtersSections, filtersSections,
sliderStates,
initFilters, initFilters,
sliderGroups, getChangedFilters
getChangedFilters,
sliderGroupsOptions
} from '../../managers/stationFilterManager'; } from '../../managers/stationFilterManager';
import { StationFilterSection } from '../../managers/stationFilterManager'; import { StationFilterSection } from '../../managers/stationFilterManager';
@@ -204,15 +206,14 @@ import { watch } from 'vue';
const STORAGE_KEY = 'options_saved'; const STORAGE_KEY = 'options_saved';
export default defineComponent({ export default defineComponent({
components: { FilterOption, FilterSlider }, components: { FilterOption },
mixins: [keyMixin, routerMixin], mixins: [keyMixin, routerMixin],
data: () => ({ data: () => ({
saveOptions: false, saveOptions: false,
filtersSections, filtersSections,
sliderGroups, sliderStates,
sliderGroupsOptions,
minimumHours: 0, minimumHours: 0,
@@ -515,7 +516,7 @@ h3.hours-section-header {
.section-filters { .section-filters {
display: grid; display: grid;
grid-template-columns: repeat(auto-fill, minmax(190px, 1fr)); grid-template-columns: repeat(3, 1fr);
gap: 0.5em; gap: 0.5em;
margin: 1em 0; margin: 1em 0;
} }
@@ -527,11 +528,9 @@ h3.hours-section-header {
-moz-user-select: none; -moz-user-select: none;
span { span {
display: flex;
justify-content: center;
align-items: center;
height: 100%;
cursor: pointer; cursor: pointer;
display: inline-block;
width: 100%;
text-align: center; text-align: center;
padding: 0.25em; padding: 0.25em;
font-weight: bold; font-weight: bold;
@@ -589,34 +588,112 @@ h3.hours-section-header {
} }
} }
.card_sliders { .slider {
margin-top: 1em;
}
.option-slider {
display: grid; display: grid;
grid-template-columns: 1fr 50px 1fr;
align-items: center; align-items: center;
grid-template-columns: 250px 100px 1fr;
gap: 0.25em; gap: 0.25em;
min-height: 35px;
margin-bottom: 1em; margin-bottom: 1em;
}
.slider-value { &-value {
color: var(--clr-primary); color: var(--clr-primary);
padding: 0.1em 0.2em; padding: 0.1em 0.2em;
text-align: center; text-align: center;
font-weight: bold; }
&-input {
-webkit-appearance: none;
appearance: none;
background: none;
border: none;
outline: none;
min-width: 25%;
&:focus-visible ~ * {
color: gold;
}
&::-webkit-slider-thumb {
-webkit-appearance: none;
appearance: none;
height: 20px;
width: 20px;
margin-top: -7px;
border-radius: 50%;
background: white;
border: 3px solid var(--clr-primary);
background-color: #333;
@include responsive.smallScreen {
width: 15px;
height: 15px;
margin-top: -5px;
border: 3px solid var(--clr-primary);
}
}
&::-moz-range-thumb {
height: 1em;
width: 1em;
border-radius: 50%;
background: white;
border: 4px solid var(--clr-primary);
cursor: pointer;
@include responsive.smallScreen {
width: 1em;
height: 1em;
border: 3px solid var(--clr-primary);
}
}
&::-webkit-slider-runnable-track {
width: 100%;
height: 5px;
cursor: pointer;
background: #ffffff;
border-radius: 1em;
}
&::-moz-range-track {
width: 100%;
height: 5px;
cursor: pointer;
background: #ffffff;
border-radius: 1em;
}
&::-ms-track {
width: 100%;
height: 5px;
cursor: pointer;
background: #ffffff;
border-radius: 1em;
}
}
} }
@include responsive.smallScreen { @include responsive.smallScreen {
.option-slider { .slider {
grid-template-columns: 1fr; display: flex;
} flex-wrap: wrap;
justify-content: center;
.slider-content { &-input {
text-align: center; width: 90%;
}
&-content {
text-align: center;
}
} }
.card_controls > button > p { .card_controls > button > p {
@@ -278,10 +278,6 @@ export default defineComponent({
color: #ccc; color: #ccc;
} }
.dropdown_wrapper {
top: 2.5em;
}
@include responsive.smallScreen { @include responsive.smallScreen {
.stats-title { .stats-title {
text-align: center; text-align: center;
@@ -290,9 +286,5 @@ export default defineComponent({
.filter-button > span { .filter-button > span {
display: none; display: none;
} }
.no-data {
text-align: center;
}
} }
</style> </style>
+5 -26
View File
@@ -1,5 +1,5 @@
<template> <template>
<section class="station_table" @scroll="onScroll" ref="tableRef"> <section class="station_table">
<Loading <Loading
v-if="apiStore.dataStatuses.connection == Status.Loading && filteredStationList.length == 0" v-if="apiStore.dataStatuses.connection == Status.Loading && filteredStationList.length == 0"
/> />
@@ -131,16 +131,7 @@
<td class="station-dispatcher-name"> <td class="station-dispatcher-name">
<span v-if="station.onlineInfo?.dispatcherName"> <span v-if="station.onlineInfo?.dispatcherName">
<b <b
v-if="isCreator(station.onlineInfo.dispatcherName)" v-if="apiStore.donatorsData.includes(station.onlineInfo.dispatcherName)"
data-tooltip-type="CreatorTooltip"
:data-tooltip-content="$t('donations.creator-message')"
>
<img src="/images/icon-creator.png" alt="creator icon" />
<span class="text--creator">&nbsp;{{ station.onlineInfo.dispatcherName }}</span>
</b>
<b
v-else-if="apiStore.donatorsData.includes(station.onlineInfo.dispatcherName)"
data-tooltip-type="DonatorTooltip" data-tooltip-type="DonatorTooltip"
:data-tooltip-content="$t('donations.dispatcher-message')" :data-tooltip-content="$t('donations.dispatcher-message')"
> >
@@ -362,7 +353,6 @@ import { ActiveSorter, HeadIdsType, headIconsIds, headIds } from './typings';
import { filterStations, sortStations } from './utils'; import { filterStations, sortStations } from './utils';
import { getLanguageNameById } from '../../utils/languageUtils'; import { getLanguageNameById } from '../../utils/languageUtils';
import FlagIcon from '../Global/FlagIcon.vue'; import FlagIcon from '../Global/FlagIcon.vue';
import { isCreator } from '../../utils/userUtils';
export default defineComponent({ export default defineComponent({
emits: ['toggleDonationCard'], emits: ['toggleDonationCard'],
@@ -373,9 +363,7 @@ export default defineComponent({
data: () => ({ data: () => ({
headIconsIds, headIconsIds,
headIds, headIds,
scrollTop: 0, getChangedFilters
getChangedFilters,
isCreator
}), }),
setup() { setup() {
@@ -403,10 +391,6 @@ export default defineComponent({
}; };
}, },
activated() {
(this.$refs['tableRef'] as HTMLElement).scrollTop = this.scrollTop;
},
methods: { methods: {
getSceneryRoute(station: Station) { getSceneryRoute(station: Station) {
this.$router.push({ this.$router.push({
@@ -447,10 +431,6 @@ export default defineComponent({
})); }));
return JSON.stringify(usersTrains); return JSON.stringify(usersTrains);
},
onScroll(e: Event) {
this.scrollTop = (e.target as HTMLElement).scrollTop;
} }
} }
}); });
@@ -598,7 +578,6 @@ tbody tr {
.station-name { .station-name {
font-weight: bold; font-weight: bold;
max-width: 200px; max-width: 200px;
padding: 0.25em;
&.default { &.default {
color: var(--clr-primary); color: var(--clr-primary);
@@ -633,8 +612,8 @@ tbody tr {
.station-dispatcher-name { .station-dispatcher-name {
img { img {
max-height: 1.3em; max-width: 1.35em;
vertical-align: text-top; vertical-align: text-bottom;
} }
} }
+21 -33
View File
@@ -120,40 +120,28 @@ function filterSliderValues(filters: Record<string, any>, generalInfo: StationGe
const otherAvailability = const otherAvailability =
availability == 'nonPublic' || availability == 'unavailable' || availability == 'abandoned'; availability == 'nonPublic' || availability == 'unavailable' || availability == 'abandoned';
if (filters['minLevel'] > reqLevel + (otherAvailability ? 1 : 0)) return true; const internalRoutes = routes.all.filter((r) => r.isInternal && !r.isRouteSBL && !r.hidden);
if (filters['maxLevel'] < reqLevel + (otherAvailability ? 1 : 0)) return true;
if (filters['minVmax'] > routes.maxRouteSpeed) return true;
if (filters['maxVmax'] < routes.minRouteSpeed) return true;
if (filters['oneWay'] && routes.singleOtherNames.length > 0) return true; return (
if (filters['oneWayCatenary'] && routes.singleElectrifiedNames.length > 0) return true; filters['minLevel'] > reqLevel + (otherAvailability ? 1 : 0) ||
if (filters['twoWay'] && routes.doubleOtherNames.length > 0) return true; filters['maxLevel'] < reqLevel + (otherAvailability ? 1 : 0) ||
if (filters['twoWayCatenary'] && routes.doubleElectrifiedNames.length > 0) return true; filters['minVmax'] > routes.maxRouteSpeed ||
filters['maxVmax'] < routes.minRouteSpeed ||
if (filters['minOneWay'] > routes.singleOtherNames.length) return true; (filters['no-1track'] && routes.single.length != 0) ||
if (filters['maxOneWay'] < routes.singleOtherNames.length) return true; (filters['no-2track'] && routes.double.length != 0) ||
if (filters['minOneWayCatenary'] > routes.singleElectrifiedNames.length) return true; filters['minOneWayCatenary'] > routes.singleElectrifiedNames.length ||
if (filters['maxOneWayCatenary'] < routes.singleElectrifiedNames.length) return true; filters['minOneWay'] > routes.singleOtherNames.length ||
if (filters['minTwoWay'] > routes.doubleOtherNames.length) return true; filters['minTwoWayCatenary'] > routes.doubleElectrifiedNames.length ||
if (filters['maxTwoWay'] < routes.doubleOtherNames.length) return true; filters['minTwoWay'] > routes.doubleOtherNames.length ||
if (filters['minTwoWayCatenary'] > routes.doubleElectrifiedNames.length) return true; filters['minOneWayCatenaryInt'] >
if (filters['maxTwoWayCatenary'] < routes.doubleElectrifiedNames.length) return true; internalRoutes.filter((r) => r.routeTracks == 1 && r.isElectric == true).length ||
filters['minOneWayInt'] >
if (filters['oneWayInt'] && routes.singleOtherInternalNames.length > 0) return true; internalRoutes.filter((r) => r.routeTracks == 1 && r.isElectric == false).length ||
if (filters['oneWayCatenaryInt'] && routes.singleElectrifiedInternalNames.length > 0) return true; filters['minTwoWayCatenaryInt'] >
if (filters['twoWayInt'] && routes.doubleOtherInternalNames.length > 0) return true; internalRoutes.filter((r) => r.routeTracks == 2 && r.isElectric == true).length ||
if (filters['twoWayCatenaryInt'] && routes.doubleElectrifiedInternalNames.length > 0) return true; filters['minTwoWayInt'] >
internalRoutes.filter((r) => r.routeTracks == 2 && r.isElectric == false).length
// Internal routes );
if (filters['minOneWayInt'] > routes.singleOtherInternalNames.length) return true;
if (filters['maxOneWayInt'] < routes.singleOtherInternalNames.length) return true;
if (filters['minOneWayCatenaryInt'] > routes.singleElectrifiedInternalNames.length) return true;
if (filters['maxOneWayCatenaryInt'] < routes.singleElectrifiedInternalNames.length) return true;
if (filters['minTwoWayInt'] > routes.doubleOtherInternalNames.length) return true;
if (filters['maxTwoWayInt'] < routes.doubleOtherInternalNames.length) return true;
if (filters['minTwoWayCatenaryInt'] > routes.doubleElectrifiedInternalNames.length) return true;
if (filters['maxTwoWayCatenaryInt'] < routes.doubleElectrifiedInternalNames.length) return true;
} }
function filterInputValues(filters: Record<string, any>, generalInfo: StationGeneralInfo) { function filterInputValues(filters: Record<string, any>, generalInfo: StationGeneralInfo) {
-36
View File
@@ -1,36 +0,0 @@
<template>
<div class="tooltip-content">
<img src="/images/icon-creator.png" alt="creator icon" />
<b class="text--creator">&nbsp;{{ tooltipStore.content }}</b>
</div>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import { useTooltipStore } from '../../store/tooltipStore';
export default defineComponent({
data() {
return {
tooltipStore: useTooltipStore()
};
}
});
</script>
<style lang="scss" scoped>
.tooltip-content {
padding: 0.5em;
border-radius: 0.25em;
width: 100%;
background-color: #333;
box-shadow: 0 0 10px 2px #aaa;
}
img {
vertical-align: text-bottom;
height: 1.25em;
}
</style>
+8 -3
View File
@@ -1,7 +1,7 @@
<template> <template>
<div class="tooltip-content"> <div class="tooltip-content">
<img src="/images/icon-diamond.svg" alt="donator diamond icon" /> <img src="/images/icon-diamond.svg" alt="donator diamond icon" />
<b class="text--donator">&nbsp;{{ tooltipStore.content }}</b> <span>{{ tooltipStore.content }}</span>
</div> </div>
</template> </template>
@@ -20,6 +20,11 @@ export default defineComponent({
<style lang="scss" scoped> <style lang="scss" scoped>
.tooltip-content { .tooltip-content {
display: flex;
align-items: center;
gap: 0.5em;
flex-wrap: wrap;
padding: 0.5em; padding: 0.5em;
border-radius: 0.25em; border-radius: 0.25em;
@@ -30,7 +35,7 @@ export default defineComponent({
} }
img { img {
vertical-align: text-bottom; vertical-align: middle;
height: 1.25em; height: 1em;
} }
</style> </style>
+2 -4
View File
@@ -8,13 +8,12 @@
import { defineComponent } from 'vue'; import { defineComponent } from 'vue';
import { useTooltipStore } from '../../store/tooltipStore'; import { useTooltipStore } from '../../store/tooltipStore';
import DonatorTooltip from './DonatorTooltip.vue'; import DonatorTooltip from './DonatorTooltip.vue';
import CreatorTooltip from './CreatorTooltip.vue';
import VehiclePreviewTooltip from './VehiclePreviewTooltip.vue'; import VehiclePreviewTooltip from './VehiclePreviewTooltip.vue';
import BaseTooltip from './BaseTooltip.vue'; import BaseTooltip from './BaseTooltip.vue';
import SpawnsTooltip from './SpawnsTooltip.vue'; import SpawnsTooltip from './SpawnsTooltip.vue';
import UsersTooltip from './UsersTooltip.vue'; import UsersTooltip from './UsersTooltip.vue';
import HtmlTooltip from './HtmlTooltip.vue'; import HtmlTooltip from './HtmlTooltip.vue';
import TrainInfoTooltip from './TrainInfoTooltip.vue'; import TrainInfoTooltip from "./TrainInfoTooltip.vue";
const BOX_PADDING_PX = 20; const BOX_PADDING_PX = 20;
@@ -26,8 +25,7 @@ export default defineComponent({
SpawnsTooltip, SpawnsTooltip,
UsersTooltip, UsersTooltip,
HtmlTooltip, HtmlTooltip,
TrainInfoTooltip, TrainInfoTooltip
CreatorTooltip
}, },
data() { data() {
+8 -15
View File
@@ -56,21 +56,12 @@
</b> </b>
<b <b
v-if="isCreator(train.driverName)" v-if="apiStore.donatorsData.includes(train.driverName)"
data-tooltip-type="CreatorTooltip"
:data-tooltip-content="$t('donations.creator-message')"
>
<img src="/images/icon-creator.png" alt="creator icon" />
<span class="text--creator">&nbsp;{{ train.driverName }}</span>
</b>
<b
v-else-if="apiStore.donatorsData.includes(train.driverName)"
data-tooltip-type="DonatorTooltip" data-tooltip-type="DonatorTooltip"
:data-tooltip-content="$t('donations.driver-message')" :data-tooltip-content="$t('donations.driver-message')"
> >
<span class="text--donator">{{ train.driverName }}&nbsp;</span>
<img src="/images/icon-diamond.svg" alt="donator diamond icon" /> <img src="/images/icon-diamond.svg" alt="donator diamond icon" />
<span class="text--donator">&nbsp;{{ train.driverName }}</span>
</b> </b>
<span v-else>{{ train.driverName }}</span> <span v-else>{{ train.driverName }}</span>
@@ -213,7 +204,6 @@ import trainCategoryMixin from '../../mixins/trainCategoryMixin';
import ProgressBar from '../Global/ProgressBar.vue'; import ProgressBar from '../Global/ProgressBar.vue';
import StockList from '../Global/StockList.vue'; import StockList from '../Global/StockList.vue';
import FlagIcon from '../Global/FlagIcon.vue'; import FlagIcon from '../Global/FlagIcon.vue';
import { isCreator } from '../../utils/userUtils';
export default defineComponent({ export default defineComponent({
mixins: [trainInfoMixin, styleMixin, trainCategoryMixin], mixins: [trainInfoMixin, styleMixin, trainCategoryMixin],
@@ -232,8 +222,7 @@ export default defineComponent({
data() { data() {
return { return {
store: useMainStore(), store: useMainStore(),
apiStore: useApiStore(), apiStore: useApiStore()
isCreator
}; };
}, },
@@ -289,6 +278,8 @@ export default defineComponent({
display: flex; display: flex;
flex-direction: column; flex-direction: column;
font-size: 1em; font-size: 1em;
gap: 0.25em;
background-color: #1a1a1a; background-color: #1a1a1a;
gap: 0.5em; gap: 0.5em;
} }
@@ -367,6 +358,8 @@ export default defineComponent({
.status-badges { .status-badges {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
margin-left: 0.25em;
gap: 0.25em; gap: 0.25em;
img { img {
@@ -379,7 +372,7 @@ export default defineComponent({
align-items: center; align-items: center;
flex-wrap: wrap; flex-wrap: wrap;
gap: 0.5em; gap: 0.5em;
padding: 0.25em 0; padding: 0.5em 0;
} }
.progress-distance { .progress-distance {
@@ -210,10 +210,6 @@ export default defineComponent({
@use '../../styles/dropdown'; @use '../../styles/dropdown';
@use '../../styles/dropdown-filters'; @use '../../styles/dropdown-filters';
.dropdown_wrapper {
top: 2.5em;
}
.search_content > div { .search_content > div {
margin: 0.5em auto; margin: 0.5em auto;
} }
+1 -2
View File
@@ -250,10 +250,9 @@ h3 {
.dropdown_wrapper { .dropdown_wrapper {
max-width: 600px; max-width: 600px;
top: 2.5em;
} }
@include responsive.smallScreen { @include responsive.smallScreen{
.no-data { .no-data {
text-align: center; text-align: center;
} }
+1 -1
View File
@@ -4,7 +4,7 @@
<TrainInfo :train="train" /> <TrainInfo :train="train" />
<div class="train-stats"> <div class="train-stats">
<StockList :trainStockList="train.stockList" :tractionOnly="true" :showPreviews="true" /> <StockList :trainStockList="train.stockList" :tractionOnly="true" />
<div> <div>
<span>{{ train.speed }}km/h</span> <span>{{ train.speed }}km/h</span>
-7
View File
@@ -35,10 +35,3 @@ export function dateToLocaleString(date: Date, dateOptions: Intl.DateTimeFormatO
return date.toLocaleString(locale.value == 'pl' ? 'pl-PL' : 'en-GB', dateOptions); return date.toLocaleString(locale.value == 'pl' ? 'pl-PL' : 'en-GB', dateOptions);
} }
export function timestampToTimeString(timestamp: number) {
return new Date(timestamp).toLocaleTimeString('pl-PL', {
hour: '2-digit',
minute: '2-digit'
});
}
-23
View File
@@ -1,23 +0,0 @@
export class HttpClient {
constructor(private readonly baseURL: string) {}
async get<T>(url: string, params?: Record<string, any>): Promise<T> {
const absoluteURL = new URL(this.baseURL + '/' + url);
if (params) {
Object.keys(params).forEach((key) => {
if (params[key] === undefined) return;
absoluteURL.searchParams.append(key, params[key]);
});
}
const data = await fetch(absoluteURL);
if (!data.ok) {
throw new Error(`Cannot fetch ${absoluteURL}: ${data.statusText}`);
}
return data.json();
}
}
+2 -36
View File
@@ -9,43 +9,9 @@ const i18n = createI18n({
warnHtmlMessage: false, warnHtmlMessage: false,
fallbackLocale: 'pl', fallbackLocale: 'pl',
pluralizationRules: {
en: {
ordinal: (ctx: { named: (arg0: string) => number }) => {
const number = ctx.named('count');
const suffixes: Record<number, string> = {
1: 'st',
2: 'nd',
3: 'rd'
};
const suffix = suffixes[number % 10] || 'th';
return `${number}${suffix}`;
}
}
},
messages: { messages: {
en: { en: enLang,
...enLang, pl: plLang
ordinal: (ctx: { named: (arg0: string) => number }) => {
const number = ctx.named('count');
const suffixes: Record<number, string> = {
1: 'st',
2: 'nd',
3: 'rd'
};
const suffix = suffixes[number % 10] || 'th';
return `${number}${suffix}`;
}
},
pl: {
...plLang,
ordinal: '{count}.'
}
}, },
enableLegacy: false enableLegacy: false
}); });
+31 -63
View File
@@ -23,6 +23,15 @@
"bottom-text": "Enjoy!\n~Spythere", "bottom-text": "Enjoy!\n~Spythere",
"button-confirm": "Start using the app!" "button-confirm": "Start using the app!"
}, },
"migrate-info": {
"tooltip-content": "Information about migration of\nStacjownik site!",
"header-text": "Attention!",
"paragraph-1-html": "Due to the growing interest in Stacjownik and other applications I have made, <b>as of January 1, 2026, Stacjownik will be <u>permanently moved</u> to a new dedicated domain:</b>",
"paragraph-2-link-text": "https://stacjownik-td2.spythere.eu",
"paragraph-3-text": "This website will no longer receive future updates and after the New Year it will only redirect to the address above.",
"paragraph-4-italic-text": "\"Why are you messing this up? It's been fine for so long!\"",
"paragraph-4-html": "<i>\"Why are you messing this up? It's been fine for so long!\"</i> <br /> The change is mainly caused by the growing website interest and exceeding the free limit plan of the current Google hosting, which forces additional fees for each use of the service above a certain threshold (or otherwise blocks access to it). By moving the site to a dedicated domain (which has already been purchased and is maintained with the financial help of <span class=\"text--donator\">Supporters</span>), I will get rid of unnecessary expenses for a large corporation that can shut down my application at any given time."
},
"donations": { "donations": {
"button-title": "TOSS A COIN", "button-title": "TOSS A COIN",
"header": "Toss a coin to Stacjownik!", "header": "Toss a coin to Stacjownik!",
@@ -42,8 +51,7 @@
"action-paypal": "DONATE WITH PAYPAL", "action-paypal": "DONATE WITH PAYPAL",
"action-buycoffee": "BUY ME A COFFEE!", "action-buycoffee": "BUY ME A COFFEE!",
"dispatcher-message": "Dispatcher supporting the Stacjownik project!", "dispatcher-message": "Dispatcher supporting the Stacjownik project!",
"driver-message": "Driver supporting the Stacjownik project!", "driver-message": "Driver supporting the Stacjownik project!"
"creator-message": "Creator of the Stacjownik project"
}, },
"warnings": { "warnings": {
"TWR": "Train with high risk cargo", "TWR": "Train with high risk cargo",
@@ -57,7 +65,7 @@
"refresh": "REFRESH" "refresh": "REFRESH"
}, },
"update": { "update": {
"title": "Stacjownik has been updated!", "title": "Stacjownik update!",
"confirm": "ROGER THAT!", "confirm": "ROGER THAT!",
"no-data": "No data about the latest app update has been found", "no-data": "No data about the latest app update has been found",
"info-1": "This changelog will be available to see once again after clicking the version number in the footer", "info-1": "This changelog will be available to see once again after clicking the version number in the footer",
@@ -200,7 +208,6 @@
"search-date-from": "Date (UTC+2 / CEST)", "search-date-from": "Date (UTC+2 / CEST)",
"search-date-to": "Date (UTC+2 / CEST)", "search-date-to": "Date (UTC+2 / CEST)",
"select-categoryCode": "Train category", "select-categoryCode": "Train category",
"search-headUnit": "Traction unit (e.g. EP09, ET22-401)",
"sort-mass": "mass", "sort-mass": "mass",
"sort-speed": "speed", "sort-speed": "speed",
"sort-length": "length", "sort-length": "length",
@@ -251,9 +258,7 @@
"blockades": "BLOCK SIGNALLING", "blockades": "BLOCK SIGNALLING",
"status": "ONLINE STATUS", "status": "ONLINE STATUS",
"timetables": "ACTIVE TIMETABLES", "timetables": "ACTIVE TIMETABLES",
"spawns": "OPEN SPAWNS", "spawns": "OPEN SPAWNS"
"externalRoutes": "EXTERNAL ROUTES",
"internalRoutes": "INTERNAL ROUTES"
}, },
"changed-filters-count": "Changed filters:", "changed-filters-count": "Changed filters:",
"no-changed-filters": "No changed filters", "no-changed-filters": "No changed filters",
@@ -297,27 +302,19 @@
"withoutActiveTimetables": "NO ACTIVE", "withoutActiveTimetables": "NO ACTIVE",
"junction": "JUNCTIONS", "junction": "JUNCTIONS",
"nonJunction": "OTHER", "nonJunction": "OTHER",
"oneWay": "OTHER SINGLE TRACK",
"oneWayCatenary": "CATENARY SINGLE TRACK",
"twoWayCatenary": "CATENARY DOUBLE TRACK",
"twoWay": "OTHER DOUBLE TRACK",
"oneWayCatenaryInt": "CATENARY SINGLE TRACK",
"oneWayInt": "OTHER SINGLE TRACK",
"twoWayCatenaryInt": "CATENARY DOUBLE TRACK",
"twoWayInt": "OTHER DOUBLE TRACK",
"sliders": { "sliders": {
"vMax": "ROUTE SPEED", "minLevel": "MIN. REQUIRED DISPATCHER LEVEL",
"level": "REQUIRED DISPATCHER LEVEL", "maxLevel": "MAX. REQUIRED DISPATCHER LEVEL",
"routeOneWay": "SINGLE TRACK ROUTES (OTHER)", "minVmax": "MIN. SCENERY ROUTE SPEED",
"routeOneWayCatenary": "SINGLE TRACK ROUTES (CATENARY)", "maxVmax": "MAX. SCENERY ROUTE SPEED",
"routeTwoWayCatenary": "DOUBLE TRACK ROUTES (CATENARY)", "minOneWayCatenary": "MIN. CATENARY SINGLE TRACK ROUTES",
"routeTwoWay": "DOUBLE TRACK ROUTES (OTHER)", "minOneWay": "MIN. OTHER SINGLE TRACK ROUTES",
"routeOneWayInternalCatenary": "INTERNAL SINGLE TRACK ROUTES (CATENARY)", "minTwoWayCatenary": "MIN. CATENARY DOUBLE TRACK ROUTES",
"routeOneWayInternal": "INTERNAL SINGLE TRACK ROUTES (OTHER)", "minTwoWay": "MIN. OTHER DOUBLE TRACK ROUTES",
"routeTwoWayInternalCatenary": "INTERNAL DOUBLE TRACK ROUTES (CATENARY)", "minOneWayCatenaryInt": "MIN. INTERNAL CATENARY SINGLE TRACK ROUTES",
"routeTwoWayInternal": "INTERNAL DOUBLE TRACK ROUTES (OTHER)" "minOneWayInt": "MIN. INTERNAL OTHER SINGLE TRACK ROUTES",
"minTwoWayCatenaryInt": "MIN. INTERNAL CATENARY DOUBLE TRACK ROUTES",
"minTwoWayInt": "MIN. INTERNAL OTHER DOUBLE TRACK ROUTES"
}, },
"sceneries-placeholder": "Search for scenery", "sceneries-placeholder": "Search for scenery",
"line-numbers-placeholder": "Line numbers (separated by commas)", "line-numbers-placeholder": "Line numbers (separated by commas)",
@@ -328,6 +325,7 @@
"now": "NOW", "now": "NOW",
"hour": "h", "hour": "h",
"no-limit": "NO LIMIT", "no-limit": "NO LIMIT",
"include-selected": "INCLUDE SELECTED",
"save": "REMEMBER FILTERS", "save": "REMEMBER FILTERS",
"reset": "RESET FILTERS", "reset": "RESET FILTERS",
"close": "CLOSE FILTERS" "close": "CLOSE FILTERS"
@@ -439,25 +437,15 @@
"driver-not-found-others": "Player {driver} is online as:", "driver-not-found-others": "Player {driver} is online as:",
"driver-not-found-return": "RETURN TO THE MAIN SITE", "driver-not-found-return": "RETURN TO THE MAIN SITE",
"stock-copy": "COPY THE STOCK", "stock-copy": "COPY THE STOCK",
"number-propositions": "NUMBER & WARNINGS SUGGESTIONS", "number-propositions": "PROPOSE NUMBER",
"stock-clipboard-success": "Successfully copied the railway stock in a text form to your clipboard!", "stock-clipboard-success": "Successfully copied the railway stock in a text form to your clipboard!",
"stock-clipboard-failure": "Oops! Something happened and the railway stock couldn't be copied to your clipboard! :/", "stock-clipboard-failure": "Oops! Something happened and the railway stock couldn't be copied to your clipboard! :/",
"number-propositions-header": "Generate number examples for a train category:", "number-propositions-header": "Generate number examples for selected category:",
"number-propositions-third-number": "Third digit:", "number-propositions-third-number": "Third digit:",
"number-propositions-last-nums": "{count} last digits from the range of:", "number-propositions-last-nums": "{count} last digits from the range of:",
"number-propositions-title": "Propositions:", "number-propositions-title": "Propositions:",
"number-propositions-empty": "No propositions available for the chosen category! :/" "number-propositions-empty": "No propositions available for the chosen category! :/"
}, },
"cargo-warnings": {
"title": "Additional cargo warnings:",
"pn-innofreight": "PN: Innofreight C45: exceeded gauge",
"twr-un1965": "TWR: UN1965 (LPG)",
"tn-un1965": "TN: unclean tanks after UN1965",
"tn-un1202": "TN: UN1202 (diesel fuel)",
"tn-un1202-empty": "TN: unclean tanks after UN1202",
"pn-military": "PN: military transport",
"pn-edk80": "PN: EDK80 railway crane"
},
"train-stats": { "train-stats": {
"stats-button": "STATISTICS", "stats-button": "STATISTICS",
"title": "ONLINE TRAINS STATS", "title": "ONLINE TRAINS STATS",
@@ -570,7 +558,7 @@
"no-users": "NO ACTIVE PLAYERS", "no-users": "NO ACTIVE PLAYERS",
"no-spawns": "NO OPEN SPAWNS", "no-spawns": "NO OPEN SPAWNS",
"no-scenery": "Oops! This scenery doesn't exist!", "no-scenery": "Oops! This scenery doesn't exist!",
"return-btn": "BACK TO SCENERIES", "return-btn": "BACK TO THE LAST SITE",
"history-btn": "View the dispatcher history", "history-btn": "View the dispatcher history",
"info-btn": "Return to the scenery view", "info-btn": "Return to the scenery view",
"authors-title": "Scenery author | Scenery authors", "authors-title": "Scenery author | Scenery authors",
@@ -580,14 +568,10 @@
"additional-tools-title": "Additional tools", "additional-tools-title": "Additional tools",
"one-way-routes": "Single track routes", "one-way-routes": "Single track routes",
"two-way-routes": "Double track routes", "two-way-routes": "Double track routes",
"routes-hidden": "Hidden internal routes",
"no-data": "No available data about this scenery", "no-data": "No available data about this scenery",
"option-active-timetables": "Active timetables", "option-active-timetables": "Active timetables",
"option-timetables-history": "Timetables history PL1", "option-timetables-history": "Timetables history PL1",
"option-dispatchers-history": "Dispatchers history PL1", "option-dispatchers-history": "Dispatchers history PL1",
"option-top-list": "Scenery records",
"btn-show-timetable-thumbnails": "Show rolling stock thumbnails",
"btn-hide-timetable-thumbnails": "Hide rolling stock thumbnails",
"timetable-includesScenery": "ALL TIMETABLES", "timetable-includesScenery": "ALL TIMETABLES",
"timetable-via": "PASSES THROUGH", "timetable-via": "PASSES THROUGH",
"timetable-issuedFrom": "BEGINS HERE", "timetable-issuedFrom": "BEGINS HERE",
@@ -598,30 +582,14 @@
"dispatcher-rate": "Rate:", "dispatcher-rate": "Rate:",
"dispatcher-status-changes": "Status changes:", "dispatcher-status-changes": "Status changes:",
"req-level": "all dispatcher levels | dispatcher level {lvl} required | dispatcher level {lvl} required", "req-level": "all dispatcher levels | dispatcher level {lvl} required | dispatcher level {lvl} required",
"history-list-empty": "No saved scenery history!", "history-list-empty": "No recorded scenery history!",
"forum-topic": "Scenery's forum topic", "forum-topic": "Official {name} forum topic",
"gnr-link": "Train orders generator", "gnr-link": "Train orders generator",
"pragotron-link": "Timetable pallet board", "pragotron-link": "Timetable pallet board",
"tablice-link": "Timetable summary board <br> (by Thundo)", "tablice-link": "Timetable summary board <br> (by Thundo)",
"bottom-info": "Show full history in the Journal tab", "bottom-info": "Show full history in the Journal tab",
"btn-show-internal-routes": "Show internal routes", "btn-show-internal-routes": "Show internal routes",
"btn-hide-internal-routes": "Hide internal routes", "btn-hide-internal-routes": "Hide internal routes"
"top-list": {
"header": "RECORDS ON THE SCENERY (PL1)",
"mode-dutyCount": "DUTIES",
"mode-dispatcherRating": "RATING",
"mode-dutyDuration": "DUTY DURATION",
"scope-name": "GENERAL",
"scope-hash": "CURRENT HASH",
"place": "place",
"dispatcher-rating": "Rating: {n}",
"duty-count": "No duties | 1 duty | Duties: {n}",
"duration": "Duration:",
"no-data-general": "No best scores for this scenery on the PL1 server!",
"no-data-current-hash": "No best scores for the current scenery hash on the PL1 server!"
}
}, },
"availability": { "availability": {
"title": "Availability", "title": "Availability",
+29 -62
View File
@@ -23,6 +23,14 @@
"bottom-text": "Miłego korzystania\n~Spythere", "bottom-text": "Miłego korzystania\n~Spythere",
"button-confirm": "Zacznij korzystać z aplikacji!" "button-confirm": "Zacznij korzystać z aplikacji!"
}, },
"migrate-info": {
"tooltip-content": "Informacja o migracji\nstrony Stacjownika!",
"header-text": "Uwaga!",
"paragraph-1-html": "Ze względu na coraz większe zainteresowanie Stacjownikiem oraz innymi aplikacjami mojego autorstwa <b>z dniem 1 stycznia 2026r. Stacjownik zostaje <u>permamentnie przeniesiony</u> na nową dedykowaną domenę:</b>",
"paragraph-2-link-text": "https://stacjownik-td2.spythere.eu",
"paragraph-3-text": "Obecna strona nie będzie otrzymywać już przyszłych aktualizacji, a po Nowym Roku będzie jedynie przenosić na powyższy adres.",
"paragraph-4-html": "<i>\"Po co psujesz? Przecież było dobrze tyle czasu!\"</i> <br /> Zmiana podyktowana jest głównie wzrostem zainteresowania stroną i przekraczaniem darmowego limitu obecnego hostingu Google'a, który wymusza płatność za każde użycie serwisu ponad określoną wartość (lub w przeciwnym wypadku blokuje do niego dostęp). Przenosząc stronę na dedykowaną domenę (która jest już wykupiona i utrzymywana dzięki pomocy <span class=\"text--donator\">Wspierających</span>), pozbędę się niepotrzebnego wydatku dla wielkiej korporacji, która w każdej chwili może mi wyłączyć aplikację."
},
"donations": { "donations": {
"button-title": "GROSZA DAJ", "button-title": "GROSZA DAJ",
"header": "Grosza daj Stacjownikowi!", "header": "Grosza daj Stacjownikowi!",
@@ -42,8 +50,7 @@
"action-paypal": "PRZELEJ PAYPALEM", "action-paypal": "PRZELEJ PAYPALEM",
"action-buycoffee": "POSTAW KAWĘ!", "action-buycoffee": "POSTAW KAWĘ!",
"dispatcher-message": "Dyżurny wspierający projekt Stacjownika!", "dispatcher-message": "Dyżurny wspierający projekt Stacjownika!",
"driver-message": "Maszynista wspierający projekt Stacjownika!", "driver-message": "Maszynista wspierający projekt Stacjownika!"
"creator-message": "Twórca projektu Stacjownik"
}, },
"warnings": { "warnings": {
"TWR": "Pociąg z towarami niebezpiecznymi wysokiego ryzyka", "TWR": "Pociąg z towarami niebezpiecznymi wysokiego ryzyka",
@@ -57,7 +64,7 @@
"refresh": "ODŚWIEŻ" "refresh": "ODŚWIEŻ"
}, },
"update": { "update": {
"title": "Stacjownik został zaktualizowany!", "title": "Aktualizacja Stacjownika!",
"confirm": "PRZYJĄŁEM!", "confirm": "PRZYJĄŁEM!",
"no-data": "Nie znaleziono informacji o ostatnich zmianach w aplikacji", "no-data": "Nie znaleziono informacji o ostatnich zmianach w aplikacji",
"info-1": "Ten changelog będzie zawsze dostępny po kliknięciu numeru wersji w stopce strony", "info-1": "Ten changelog będzie zawsze dostępny po kliknięciu numeru wersji w stopce strony",
@@ -197,7 +204,6 @@
"search-date-from": "Data (UTC+2 / CEST)", "search-date-from": "Data (UTC+2 / CEST)",
"search-date-to": "Data (UTC+2 / CEST)", "search-date-to": "Data (UTC+2 / CEST)",
"select-categoryCode": "Kategoria pociągu", "select-categoryCode": "Kategoria pociągu",
"search-headUnit": "Pojazd trakcyjny (np. EP09, ET22-137)",
"sort-routeDistance": "kilometraż", "sort-routeDistance": "kilometraż",
"sort-allStopsCount": "stacje", "sort-allStopsCount": "stacje",
"sort-beginDate": "data", "sort-beginDate": "data",
@@ -249,9 +255,7 @@
"blockades": "BLOKADY LINIOWE", "blockades": "BLOKADY LINIOWE",
"status": "STATUS ONLINE", "status": "STATUS ONLINE",
"timetables": "AKTYWNE ROZKŁADY JAZDY", "timetables": "AKTYWNE ROZKŁADY JAZDY",
"spawns": "OTWARTE SPAWNY", "spawns": "OTWARTE SPAWNY"
"externalRoutes": "SZLAKI ZEWNĘTRZNE",
"internalRoutes": "SZLAKI WEWNĘTRZNE"
}, },
"changed-filters-count": "Zmienione filtry:", "changed-filters-count": "Zmienione filtry:",
"no-changed-filters": "Brak zmienionych filtrów", "no-changed-filters": "Brak zmienionych filtrów",
@@ -295,27 +299,19 @@
"withoutActiveTimetables": "BEZ AKTYWNYCH", "withoutActiveTimetables": "BEZ AKTYWNYCH",
"junction": "WĘZŁOWE", "junction": "WĘZŁOWE",
"nonJunction": "INNE", "nonJunction": "INNE",
"oneWay": "JEDNOTOROWE NIEZELEKTRYFIKOWANE",
"oneWayCatenary": "JEDNOTOROWE ZELEKTRYFIKOWANE",
"twoWayCatenary": "DWUTOROWE ZELEKTRYFIKOWANE",
"twoWay": "DWUTOROWE NIEZELEKTRYFIKOWANE",
"oneWayCatenaryInt": "JEDNOTOROWE ZELEKTRYFIKOWANE",
"oneWayInt": "JEDNOTOROWE NIEZELEKTRYFIKOWANE",
"twoWayCatenaryInt": "DWUTOROWE ZELEKTRYFIKOWANE",
"twoWayInt": "DWUTOROWE NIEZELEKTRYFIKOWANE",
"sliders": { "sliders": {
"vMax": "PRĘDKOŚĆ SZLAKOWA", "minLevel": "MIN. WYMAGANY POZIOM DYŻURNEGO",
"level": "WYMAGANY POZIOM DYŻURNEGO", "maxLevel": "MAKS. WYMAGANY POZIOM DYŻURNEGO",
"routeOneWay": "SZLAKI 1-TOROWE NIEZELEKTR.", "minVmax": "MIN. PRĘDKOŚĆ SZLAKOWA",
"routeOneWayCatenary": "SZLAKI 1-TOROWE ZELEKTR.", "maxVmax": "MAKS. PRĘDKOŚĆ SZLAKOWA",
"routeTwoWayCatenary": "SZLAKI 2-TOROWE ZELEKTR.", "minOneWayCatenary": "SZLAKI JEDNOTOROWE ZELEKTR. (MINIMUM)",
"routeTwoWay": "SZLAKI 2-TOROWE NIEZELEKTR.", "minOneWay": "SZLAKI JEDNOTOROWE NIEZELEKTR. (MINIMUM)",
"routeOneWayInternalCatenary": "SZLAKI WEWN. 1-TOROWE ZELEKTR.", "minTwoWayCatenary": "SZLAKI DWUTOROWE ZELEKTR. (MINIMUM)",
"routeOneWayInternal": "SZLAKI WEWN. 1-TOROWE NIEZELEKTR.", "minTwoWay": "SZLAKI DWUTOROWE NIEZELEKTR. (MINIMUM)",
"routeTwoWayInternalCatenary": "SZLAKI WEWN. 2-TOROWE ZELEKTR.", "minOneWayCatenaryInt": "SZLAKI JEDNOTOROWE ZELEKTR. WEWNĘTRZNE (MINIMUM)",
"routeTwoWayInternal": "SZLAKI WEWN. 2-TOROWE NIEZELEKTR." "minOneWayInt": "SZLAKI JEDNOTOROWE NIEZELEKTR. WEWNĘTRZNE (MINIMUM)",
"minTwoWayCatenaryInt": "SZLAKI DWUTOROWE ZELEKTR. WEWNĘTRZNE (MINIMUM)",
"minTwoWayInt": "SZLAKI DWUTOROWE NIEZELEKTR. WEWNĘTRZNE (MINIMUM)"
}, },
"sceneries-placeholder": "Wyszukaj scenerię", "sceneries-placeholder": "Wyszukaj scenerię",
"line-numbers-placeholder": "Numery linii (oddzielone przecinkami)", "line-numbers-placeholder": "Numery linii (oddzielone przecinkami)",
@@ -326,6 +322,7 @@
"now": "TERAZ", "now": "TERAZ",
"hour": " godz.", "hour": " godz.",
"no-limit": "BEZ LIMITU", "no-limit": "BEZ LIMITU",
"include-selected": "POKAŻ ZAZNACZONE",
"save": "ZAPAMIĘTAJ FILTRY", "save": "ZAPAMIĘTAJ FILTRY",
"reset": "RESETUJ FILTRY", "reset": "RESETUJ FILTRY",
"close": "ZAMKNIJ FILTRY" "close": "ZAMKNIJ FILTRY"
@@ -426,25 +423,15 @@
"driver-not-found-others": "Gracz {driver} jest online jako:", "driver-not-found-others": "Gracz {driver} jest online jako:",
"driver-not-found-return": "WRÓĆ NA STRONĘ GŁÓWNĄ", "driver-not-found-return": "WRÓĆ NA STRONĘ GŁÓWNĄ",
"stock-copy": "SKOPIUJ SKŁAD", "stock-copy": "SKOPIUJ SKŁAD",
"number-propositions": "PROPOZYCJE NUMERÓW I UWAG", "number-propositions": "ZAPROPONUJ NUMER",
"stock-clipboard-success": "Pomyślnie skopiowano skład w postaci tekstowej do schowka!", "stock-clipboard-success": "Pomyślnie skopiowano skład w postaci tekstowej do schowka!",
"stock-clipboard-failure": "Ups! Nie udało się skopiować składu do schowka! :/", "stock-clipboard-failure": "Ups! Nie udało się skopiować składu do schowka! :/",
"number-propositions-header": "Wygeneruj propozycje numerów dla pociągu kategorii:", "number-propositions-header": "Wygeneruj propozycje numerów dla kategorii pociągu:",
"number-propositions-third-number": "Trzecia cyfra:", "number-propositions-third-number": "Trzecia cyfra:",
"number-propositions-last-nums": "{count} ostatnie cyfry z przedziału:", "number-propositions-last-nums": "{count} ostatnie cyfry z przedziału:",
"number-propositions-title": "Propozycje:", "number-propositions-title": "Propozycje:",
"number-propositions-empty": "Brak propozycji dla wybranej kategorii! :/" "number-propositions-empty": "Brak propozycji dla wybranej kategorii! :/"
}, },
"cargo-warnings": {
"title": "Dodatkowe uwagi przewozowe:",
"pn-innofreight": "PN: Innofreight C45: przekroczona skrajnia",
"twr-un1965": "TWR: UN1965 (LPG)",
"tn-un1965": "TN: brudne cysterny po UN1965",
"tn-un1202": "TN: UN1202 (olej napędowy)",
"tn-un1202-empty": "TN: brudne cysterny po UN1202",
"pn-military": "PN: transport wojskowy",
"pn-edk80": "PN: żuraw kolejowy EDK80"
},
"train-stats": { "train-stats": {
"stats-button": "STATYSTYKI", "stats-button": "STATYSTYKI",
"title": "STATYSTYKI AKTYWNYCH POCIĄGÓW", "title": "STATYSTYKI AKTYWNYCH POCIĄGÓW",
@@ -556,7 +543,7 @@
"no-users": "BRAK AKTYWNYCH GRACZY", "no-users": "BRAK AKTYWNYCH GRACZY",
"no-spawns": "BRAK OTWARTYCH SPAWNÓW", "no-spawns": "BRAK OTWARTYCH SPAWNÓW",
"no-scenery": "Ups! Ta sceneria nie istnieje!", "no-scenery": "Ups! Ta sceneria nie istnieje!",
"return-btn": "POWRÓT DO SCENERII", "return-btn": "POWRÓT DO POPRZEDNIEJ STRONY",
"history-btn": "Przejdź do widoku historii dyżurnych ruchu", "history-btn": "Przejdź do widoku historii dyżurnych ruchu",
"info-btn": "Wróć do widoku scenerii", "info-btn": "Wróć do widoku scenerii",
"authors-title": "Autor scenerii | Autorzy scenerii", "authors-title": "Autor scenerii | Autorzy scenerii",
@@ -566,14 +553,10 @@
"additional-tools-title": "Dodatkowe narzędzia", "additional-tools-title": "Dodatkowe narzędzia",
"one-way-routes": "Szlaki jednotorowe", "one-way-routes": "Szlaki jednotorowe",
"two-way-routes": "Szlaki dwutorowe", "two-way-routes": "Szlaki dwutorowe",
"routes-hidden": "Ukryto szlaki wewnętrzne",
"no-data": "Brak informacji o tej scenerii", "no-data": "Brak informacji o tej scenerii",
"option-active-timetables": "Aktywne rozkłady jazdy", "option-active-timetables": "Aktywne rozkłady jazdy",
"option-timetables-history": "Historia rozkładów PL1", "option-timetables-history": "Historia rozkładów PL1",
"option-dispatchers-history": "Historia dyżurów PL1", "option-dispatchers-history": "Historia dyżurów PL1",
"option-top-list": "Rekordy scenerii",
"btn-show-timetable-thumbnails": "Pokazuj podglądy składów",
"btn-hide-timetable-thumbnails": "Ukrywaj podglądy składów",
"timetable-includesScenery": "WSZYSTKIE RJ", "timetable-includesScenery": "WSZYSTKIE RJ",
"timetable-via": "PRZEJEŻDŻA", "timetable-via": "PRZEJEŻDŻA",
"timetable-issuedFrom": "ROZPOCZYNA BIEG", "timetable-issuedFrom": "ROZPOCZYNA BIEG",
@@ -585,29 +568,13 @@
"dispatcher-status-changes": "Zmiany statusów:", "dispatcher-status-changes": "Zmiany statusów:",
"req-level": "ogólnodostępna | minimum {lvl} poziom dyżurnego | minimum {lvl} poziom dyżurnego", "req-level": "ogólnodostępna | minimum {lvl} poziom dyżurnego | minimum {lvl} poziom dyżurnego",
"history-list-empty": "Brak historii dla tej scenerii!", "history-list-empty": "Brak historii dla tej scenerii!",
"forum-topic": "Wątek scenerii", "forum-topic": "Oficjalny wątek scenerii {name}",
"gnr-link": "Generator rozkazów pisemnych", "gnr-link": "Generator rozkazów pisemnych",
"pragotron-link": "Paletowa tablica informacyjna", "pragotron-link": "Paletowa tablica informacyjna",
"tablice-link": "Tablica informacyjna zbiorcza <br> (autorstwa Thundo)", "tablice-link": "Tablica informacyjna zbiorcza <br> (autorstwa Thundo)",
"bottom-info": "Pokaż pełną historię w zakładce Dziennika", "bottom-info": "Pokaż pełną historię w zakładce Dziennika",
"btn-show-internal-routes": "Pokazuj szlaki wewnętrzne", "btn-show-internal-routes": "Pokazuj szlaki wewnętrzne",
"btn-hide-internal-routes": "Ukrywaj szlaki wewnętrzne", "btn-hide-internal-routes": "Ukrywaj szlaki wewnętrzne"
"top-list": {
"header": "REKORDY NA SCENERII (PL1)",
"mode-dutyCount": "DYŻURY",
"mode-dispatcherRating": "OCENA",
"mode-dutyDuration": "CZAS DYŻURU",
"scope-name": "OGÓLNIE",
"scope-hash": "OBECNY HASH",
"place": "miejsce",
"dispatcher-rating": "Ocena: {n}",
"duty-count": "Brak dyżurów | 1 dyżur | Dyżury: {n}",
"duration": "Czas:",
"no-data-general": "Brak zapisanych rekordów scenerii na serwerze PL1!",
"no-data-current-hash": "Brak zapisanych rekordów scenerii z obecnym hashem na serwerze PL1!"
}
}, },
"availability": { "availability": {
"title": "Dostępność", "title": "Dostępność",
+20 -118
View File
@@ -1,24 +1,5 @@
import StorageManager from './storageManager'; import StorageManager from './storageManager';
export type SliderGroup =
| 'vMax'
| 'level'
| 'routeOneWay'
| 'routeOneWayCatenary'
| 'routeOneWayInternal'
| 'routeOneWayInternalCatenary'
| 'routeTwoWay'
| 'routeTwoWayCatenary'
| 'routeTwoWayInternal'
| 'routeTwoWayInternalCatenary';
export interface SliderOptions {
id: string;
minRange: number;
maxRange: number;
step: number;
}
export const sections = [ export const sections = [
'status', 'status',
'timetables', 'timetables',
@@ -29,9 +10,7 @@ export const sections = [
'control', 'control',
'blockades', 'blockades',
'signals', 'signals',
'addons', 'addons'
'externalRoutes',
'internalRoutes'
] as const; ] as const;
export const initFilters = { export const initFilters = {
@@ -59,6 +38,9 @@ export const initFilters = {
mixed: false, mixed: false,
SBL: false, SBL: false,
PBL: false, PBL: false,
'include-selected': false,
'no-1track': false,
'no-2track': false,
free: true, free: true,
occupied: false, occupied: false,
nonPublic: false, nonPublic: false,
@@ -78,111 +60,34 @@ export const initFilters = {
onlineFromHours: 0, onlineFromHours: 0,
minLevel: 0, minLevel: 0,
maxLevel: 20, maxLevel: 20,
oneWay: false,
oneWayCatenary: false,
twoWay: false,
twoWayCatenary: false,
oneWayCatenaryInt: false,
oneWayInt: false,
twoWayInt: false,
twoWayCatenaryInt: false,
minOneWay: 0, minOneWay: 0,
minOneWayCatenary: 0, minOneWayCatenary: 0,
minOneWayCatenaryInt: 0,
minOneWayInt: 0, minOneWayInt: 0,
minOneWayCatenaryInt: 0,
minTwoWay: 0, minTwoWay: 0,
minTwoWayCatenary: 0, minTwoWayCatenary: 0,
minTwoWayInt: 0, minTwoWayInt: 0,
minTwoWayCatenaryInt: 0, minTwoWayCatenaryInt: 0,
maxOneWay: 10,
maxOneWayCatenary: 10,
maxOneWayInt: 20,
maxOneWayCatenaryInt: 20,
maxTwoWay: 10,
maxTwoWayCatenary: 10,
maxTwoWayInt: 20,
maxTwoWayCatenaryInt: 20,
authors: '', authors: '',
projects: '', projects: '',
lines: '' lines: ''
}; };
export const sliderGroups: SliderGroup[] = [ export const sliderStates = [
'vMax', { id: 'maxVmax', minRange: 0, maxRange: 200, step: 10 },
'level', { id: 'minVmax', minRange: 0, maxRange: 200, step: 10 },
'routeOneWayCatenary', { id: 'minLevel', minRange: 0, maxRange: 20, step: 1 },
'routeOneWay', { id: 'maxLevel', minRange: 0, maxRange: 20, step: 1 },
'routeTwoWayCatenary', { id: 'minOneWay', minRange: 0, maxRange: 5, step: 1 },
'routeTwoWay', { id: 'minOneWayCatenary', minRange: 0, maxRange: 5, step: 1 },
'routeOneWayInternalCatenary', { id: 'minOneWayInt', minRange: 0, maxRange: 5, step: 1 },
'routeOneWayInternal', { id: 'minOneWayCatenaryInt', minRange: 0, maxRange: 5, step: 1 },
'routeTwoWayInternalCatenary', { id: 'minTwoWay', minRange: 0, maxRange: 5, step: 1 },
'routeTwoWayInternal' { id: 'minTwoWayCatenary', minRange: 0, maxRange: 5, step: 1 },
{ id: 'minTwoWayInt', minRange: 0, maxRange: 5, step: 1 },
{ id: 'minTwoWayCatenaryInt', minRange: 0, maxRange: 5, step: 1 }
]; ];
export const sliderGroupsOptions: Record<SliderGroup, SliderOptions[]> = {
vMax: [
{ id: 'minVmax', minRange: 0, maxRange: 200, step: 20 },
{ id: 'maxVmax', minRange: 0, maxRange: 200, step: 20 }
],
level: [
{ id: 'minLevel', minRange: 0, maxRange: 20, step: 1 },
{ id: 'maxLevel', minRange: 0, maxRange: 20, step: 1 }
],
routeOneWay: [
{ id: 'minOneWay', minRange: 0, maxRange: 10, step: 1 },
{ id: 'maxOneWay', minRange: 0, maxRange: 10, step: 1 }
],
routeOneWayCatenary: [
{ id: 'minOneWayCatenary', minRange: 0, maxRange: 10, step: 1 },
{ id: 'maxOneWayCatenary', minRange: 0, maxRange: 10, step: 1 }
],
routeOneWayInternal: [
{ id: 'minOneWayInt', minRange: 0, maxRange: 20, step: 1 },
{ id: 'maxOneWayInt', minRange: 0, maxRange: 20, step: 1 }
],
routeOneWayInternalCatenary: [
{
id: 'minOneWayCatenaryInt',
minRange: 0,
maxRange: 20,
step: 1
},
{
id: 'maxOneWayCatenaryInt',
minRange: 0,
maxRange: 20,
step: 1
}
],
routeTwoWay: [
{ id: 'minTwoWay', minRange: 0, maxRange: 10, step: 1 },
{ id: 'maxTwoWay', minRange: 0, maxRange: 10, step: 1 }
],
routeTwoWayCatenary: [
{ id: 'minTwoWayCatenary', minRange: 0, maxRange: 10, step: 1 },
{ id: 'maxTwoWayCatenary', minRange: 0, maxRange: 10, step: 1 }
],
routeTwoWayInternal: [
{ id: 'minTwoWayInt', minRange: 0, maxRange: 20, step: 1 },
{ id: 'maxTwoWayInt', minRange: 0, maxRange: 20, step: 1 }
],
routeTwoWayInternalCatenary: [
{
id: 'minTwoWayCatenaryInt',
minRange: 0,
maxRange: 20,
step: 1
},
{
id: 'maxTwoWayCatenaryInt',
minRange: 0,
maxRange: 20,
step: 1
}
]
};
export type StationFilter = keyof typeof initFilters; export type StationFilter = keyof typeof initFilters;
export type StationFilterSection = (typeof sections)[number]; export type StationFilterSection = (typeof sections)[number];
@@ -207,9 +112,7 @@ export const filtersSections: Record<StationFilterSection, StationFilter[]> = {
'manual' 'manual'
], ],
blockades: ['SBL', 'PBL'], blockades: ['SBL', 'PBL'],
signals: ['modern', 'semaphores', 'mixed', 'historical'], signals: ['modern', 'semaphores', 'mixed', 'historical']
externalRoutes: ['oneWayCatenary', 'oneWay', 'twoWayCatenary', 'twoWay'],
internalRoutes: ['oneWayCatenaryInt', 'oneWayInt', 'twoWayCatenaryInt', 'twoWayInt']
}; };
export function setupFilters(currentFilters: Record<string, any>) { export function setupFilters(currentFilters: Record<string, any>) {
@@ -232,8 +135,7 @@ export function getChangedFilters(currentFilters: Record<string, any>): string[]
return ( return (
Object.keys(currentFilters).filter( Object.keys(currentFilters).filter(
(filterKey) => (filterKey) =>
currentFilters[filterKey].toString() !== currentFilters[filterKey] !== initFilters[filterKey as keyof typeof initFilters]
initFilters[filterKey as keyof typeof initFilters].toString()
) ?? [] ) ?? []
); );
} }
+37 -33
View File
@@ -2,25 +2,11 @@ import { defineStore } from 'pinia';
import { API } from '../typings/api'; import { API } from '../typings/api';
import { Status } from '../typings/common'; import { Status } from '../typings/common';
import { StationJSONData } from './typings'; import { StationJSONData } from './typings';
import { HttpClient } from '../http'; import axios, { AxiosInstance } from 'axios';
let baseURL = 'https://stacjownik.spythere.eu';
switch (import.meta.env.VITE_API_MODE) {
case 'development':
baseURL = 'http://localhost:3001';
break;
case 'mocking':
baseURL = 'http://localhost:3123';
break;
default:
break;
}
export const useApiStore = defineStore('apiStore', { export const useApiStore = defineStore('apiStore', {
state: () => ({ state: () => ({
dataStatuses: { dataStatuses: {
allData: Status.Data.Loading,
connection: Status.Data.Loading, connection: Status.Data.Loading,
sceneries: Status.Data.Loading, sceneries: Status.Data.Loading,
vehicles: Status.Data.Loading, vehicles: Status.Data.Loading,
@@ -38,13 +24,30 @@ export const useApiStore = defineStore('apiStore', {
nextUpdateTime: 0, nextUpdateTime: 0,
nextDataCheckTime: 0, nextDataCheckTime: 0,
client: new HttpClient(baseURL), client: undefined as AxiosInstance | undefined,
activeDataScheduler: undefined as number | undefined activeDataScheduler: undefined as number | undefined
}), }),
actions: { actions: {
async setupAPIData() { async setupAPIData() {
let baseURL = 'https://stacjownik.spythere.eu';
switch (import.meta.env.VITE_API_MODE) {
case 'development':
baseURL = 'http://localhost:3001';
break;
case 'mocking':
baseURL = 'http://localhost:3123';
break;
default:
break;
}
this.client = axios.create({
baseURL
});
this.connectToAPI(); this.connectToAPI();
}, },
@@ -52,21 +55,19 @@ export const useApiStore = defineStore('apiStore', {
window.requestAnimationFrame(this.updateTick); window.requestAnimationFrame(this.updateTick);
}, },
async updateTick(t: number) { updateTick(t: number) {
// Static data refresh // Static data refresh
if (t >= this.nextDataCheckTime) { if (t >= this.nextDataCheckTime) {
await Promise.all([ this.fetchDonatorsData();
this.fetchStationsGeneralInfo(), this.fetchVehiclesInfo();
this.fetchVehiclesInfo(), this.fetchStationsGeneralInfo();
this.fetchDonatorsData()
]);
this.nextDataCheckTime = t + 3600000; this.nextDataCheckTime = t + 3600000;
} }
// Active data fefresh // Active data fefresh
if (t >= this.nextUpdateTime) { if (t >= this.nextUpdateTime) {
await this.fetchActiveData(); this.fetchActiveData();
this.nextUpdateTime = t + 31000; this.nextUpdateTime = t + 31000;
} }
@@ -74,13 +75,12 @@ export const useApiStore = defineStore('apiStore', {
}, },
async fetchActiveData() { async fetchActiveData() {
if (this.dataStatuses.connection == Status.Data.Offline) return;
if (!this.activeData) this.dataStatuses.connection = Status.Data.Loading; if (!this.activeData) this.dataStatuses.connection = Status.Data.Loading;
try { try {
const response = await this.client.get<API.ActiveData.Response>('api/getActiveData'); const response = await this.client!.get<API.ActiveData.Response>('api/getActiveData');
this.activeData = response; this.activeData = response.data;
this.dataStatuses.connection = Status.Data.Loaded; this.dataStatuses.connection = Status.Data.Loaded;
} catch (error) { } catch (error) {
this.dataStatuses.connection = Status.Data.Error; this.dataStatuses.connection = Status.Data.Error;
@@ -90,9 +90,9 @@ export const useApiStore = defineStore('apiStore', {
async fetchDonatorsData() { async fetchDonatorsData() {
try { try {
const response = await this.client.get<API.Donators.Response>('api/getDonators'); const response = await this.client!.get<API.Donators.Response>('api/getDonators');
this.donatorsData = response; this.donatorsData = response.data;
} catch (error) { } catch (error) {
console.error('Ups! Wystąpił błąd podczas pobierania informacji o donatorach:', error); console.error('Ups! Wystąpił błąd podczas pobierania informacji o donatorach:', error);
} }
@@ -100,7 +100,9 @@ export const useApiStore = defineStore('apiStore', {
async fetchStationsGeneralInfo() { async fetchStationsGeneralInfo() {
try { try {
const sceneryData = await this.client.get<StationJSONData[]>(`api/getSceneries`); const sceneryData: StationJSONData[] = (
await this.client!.get<StationJSONData[]>(`api/getSceneries`)
).data;
this.dataStatuses.sceneries = Status.Data.Loaded; this.dataStatuses.sceneries = Status.Data.Loaded;
this.sceneryData = sceneryData; this.sceneryData = sceneryData;
@@ -112,10 +114,10 @@ export const useApiStore = defineStore('apiStore', {
async fetchVehiclesInfo() { async fetchVehiclesInfo() {
try { try {
const response = await this.client.get<API.VehiclesData.Response>('api/getVehiclesData'); const response = await this.client!.get<API.VehiclesData.Response>('api/getVehiclesData');
this.vehiclesData = response; this.vehiclesData = response.data;
this.dataStatuses.vehicles = response ? Status.Data.Loaded : Status.Data.Warning; this.dataStatuses.vehicles = response.data ? Status.Data.Loaded : Status.Data.Warning;
} catch (error) { } catch (error) {
this.dataStatuses.vehicles = Status.Data.Error; this.dataStatuses.vehicles = Status.Data.Error;
console.error('Ups! Wystąpił błąd podczas pobierania informacji o pojazdach:', error); console.error('Ups! Wystąpił błąd podczas pobierania informacji o pojazdach:', error);
@@ -124,7 +126,9 @@ export const useApiStore = defineStore('apiStore', {
async fetchDailyStats() { async fetchDailyStats() {
try { try {
const res = await this.client.get<API.DailyStats.Response>('api/getDailyStats'); const res: API.DailyStats.Response = await (
await this.client!.get('api/getDailyStats')
).data;
this.dailyStatsData = res; this.dailyStatsData = res;
+7 -14
View File
@@ -29,7 +29,9 @@ export const useMainStore = defineStore('mainStore', {
chosenModalTrainId: undefined, chosenModalTrainId: undefined,
modalLastClickedTarget: null, modalLastClickedTarget: null,
currentLocale: 'pl' currentLocale: 'pl',
isMigrateInfoCardOpen: false
}) as MainStoreState, }) as MainStoreState,
actions: { actions: {
@@ -391,13 +393,11 @@ export const useMainStore = defineStore('mainStore', {
const tracksKey = route.routeTracks == 2 ? 'double' : 'single'; const tracksKey = route.routeTracks == 2 ? 'double' : 'single';
const isElectric = route.isElectric; const isElectric = route.isElectric;
const routesKey: keyof StationRoutes = `${tracksKey}${ const routesKey: keyof StationRoutes = `${tracksKey}${
!isElectric ? 'Other' : 'Electrified' !isElectric ? 'Other' : 'Electrified'
}${route.isInternal ? 'Internal' : ''}Names`; }Names`;
acc[routesKey].push(route.routeName);
if (!route.isInternal) acc[routesKey].push(route.routeName);
if (route.isRouteSBL) acc['sblNames'].push(route.routeName); if (route.isRouteSBL) acc['sblNames'].push(route.routeName);
acc.minRouteSpeed = acc.minRouteSpeed =
@@ -412,21 +412,14 @@ export const useMainStore = defineStore('mainStore', {
return acc; return acc;
}, },
{ {
all: [],
single: [], single: [],
double: [],
singleElectrifiedNames: [], singleElectrifiedNames: [],
singleOtherNames: [], singleOtherNames: [],
double: [],
doubleElectrifiedNames: [], doubleElectrifiedNames: [],
doubleOtherNames: [], doubleOtherNames: [],
singleElectrifiedInternalNames: [],
singleOtherInternalNames: [],
doubleElectrifiedInternalNames: [],
doubleOtherInternalNames: [],
sblNames: [], sblNames: [],
all: [],
minRouteSpeed: 0, minRouteSpeed: 0,
maxRouteSpeed: 0 maxRouteSpeed: 0
} as StationRoutes } as StationRoutes
+1 -2
View File
@@ -9,8 +9,7 @@ export const tooltipKeys = [
'SpawnsTooltip', 'SpawnsTooltip',
'UsersTooltip', 'UsersTooltip',
'HtmlTooltip', 'HtmlTooltip',
'TrainInfoTooltip', 'TrainInfoTooltip'
'CreatorTooltip'
] as const; ] as const;
export type TooltipType = (typeof tooltipKeys)[number]; export type TooltipType = (typeof tooltipKeys)[number];
+1
View File
@@ -8,6 +8,7 @@ export interface MainStoreState {
chosenModalTrainId?: string; chosenModalTrainId?: string;
modalLastClickedTarget: EventTarget | null; modalLastClickedTarget: EventTarget | null;
currentLocale: string; currentLocale: string;
isMigrateInfoCardOpen: boolean;
} }
export interface StationJSONData { export interface StationJSONData {
+1 -1
View File
@@ -1,4 +1,4 @@
$animDuration: 120ms; $animDuration: 95ms;
$animType: ease-in-out; $animType: ease-in-out;
// List animation // List animation
+1
View File
@@ -85,6 +85,7 @@
padding: 0.1em 0.3em; padding: 0.1em 0.3em;
border-radius: 0.2em; border-radius: 0.2em;
font-weight: bold; font-weight: bold;
user-select: none;
&.twr { &.twr {
background-color: var(--clr-twr); background-color: var(--clr-twr);
+1 -1
View File
@@ -85,7 +85,7 @@ h1.option-title {
} }
} }
@include responsive.smallScreen { @include responsive.smallScreen{
h1 { h1 {
text-align: center; text-align: center;
+9 -1
View File
@@ -26,7 +26,7 @@
.dropdown_wrapper { .dropdown_wrapper {
position: absolute; position: absolute;
left: 0; left: 0;
top: 0; top: calc(100% + 0.5em);
background-color: var(--clr-bg3); background-color: var(--clr-bg3);
box-shadow: 0 0 5px 1px var(--clr-primary); box-shadow: 0 0 5px 1px var(--clr-primary);
@@ -34,8 +34,16 @@
width: 100%; width: 100%;
max-width: 550px; max-width: 550px;
max-height: 750px;
overflow: auto; overflow: auto;
padding: 1em; padding: 1em;
z-index: 100; z-index: 100;
} }
@include responsive.smallScreen {
.dropdown_wrapper {
font-size: 1.1em;
max-width: 100%;
}
}
-13
View File
@@ -217,19 +217,6 @@ ul {
text-shadow: #f050ff 0 0 10px; text-shadow: #f050ff 0 0 10px;
} }
&--creator {
color: var(--clr-primary);
color: transparent;
background: var(--clr-primary);
background: linear-gradient(90deg, gold 30%, #ffffff 70%);
background-clip: text;
-webkit-background-clip: text;
-webkit-text-fill-color: transparent;
text-shadow: gold 0 0 10px;
}
&--discord { &--discord {
color: var(--clr-donator); color: var(--clr-donator);
color: transparent; color: transparent;
+3 -2
View File
@@ -24,8 +24,8 @@
width: 100%; width: 100%;
margin: 0 auto; margin: 0 auto;
padding: 1em 0; padding: 1em 0;
position: relative;
} }
.journal_refreshed-date { .journal_refreshed-date {
@@ -57,6 +57,7 @@
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
gap: 0.5em; gap: 0.5em;
position: relative;
} }
.btn--load-data { .btn--load-data {
@@ -67,7 +68,7 @@
font-size: 1.2em; font-size: 1.2em;
} }
@include responsive.smallScreen { @include responsive.smallScreen{
.journal_top-bar { .journal_top-bar {
justify-content: center; justify-content: center;
flex-wrap: wrap; flex-wrap: wrap;
+1
View File
@@ -15,6 +15,7 @@
gap: 0.25em; gap: 0.25em;
min-width: 200px; min-width: 200px;
margin-right: 0.25em;
} }
&-input { &-input {
+1 -3
View File
@@ -253,10 +253,8 @@ export namespace API {
pn?: number; pn?: number;
tn?: number; tn?: number;
headUnitName?: string;
headUnitType?: string;
returnType?: 'all' | 'short' | 'detailed'; returnType?: 'all' | 'short' | 'detailed';
sortBy?: Journal.TimetableSorter['id']; sortBy?: Journal.TimetableSorter['id'];
} }
-6
View File
@@ -130,12 +130,6 @@ export interface StationRoutes {
singleOtherNames: string[]; singleOtherNames: string[];
doubleElectrifiedNames: string[]; doubleElectrifiedNames: string[];
doubleOtherNames: string[]; doubleOtherNames: string[];
singleElectrifiedInternalNames: string[];
singleOtherInternalNames: string[];
doubleElectrifiedInternalNames: string[];
doubleOtherInternalNames: string[];
sblNames: string[]; sblNames: string[];
minRouteSpeed: number; minRouteSpeed: number;
-3
View File
@@ -1,3 +0,0 @@
export function isCreator(name: string) {
return /(spythere|kowbojyt)/.test(name.toLowerCase());
}
+6 -8
View File
@@ -217,10 +217,9 @@ export default defineComponent({
this.scrollDataLoaded = false; this.scrollDataLoaded = false;
this.currentQueryParams['countFrom'] = this.historyList.length; this.currentQueryParams['countFrom'] = this.historyList.length;
const responseData: API.DispatcherHistory.Response = await await this.apiStore.client.get( const responseData: API.DispatcherHistory.Response = await (
`api/getDispatchers`, await this.apiStore.client!.get(`api/getDispatchers`, { params: this.currentQueryParams })
this.currentQueryParams ).data;
);
if (!responseData) return; if (!responseData) return;
@@ -277,10 +276,9 @@ export default defineComponent({
this.currentQueryParams = queryParams; this.currentQueryParams = queryParams;
try { try {
const responseData: API.DispatcherHistory.Response = await this.apiStore.client.get( const responseData: API.DispatcherHistory.Response = await (
`api/getDispatchers`, await this.apiStore.client!.get(`api/getDispatchers`, { params: this.currentQueryParams })
this.currentQueryParams ).data;
);
if (!responseData) { if (!responseData) {
this.dataStatus = Status.Data.Error; this.dataStatus = Status.Data.Error;
+14 -25
View File
@@ -173,9 +173,8 @@ export default defineComponent({
'search-issuedFrom': '', 'search-issuedFrom': '',
'search-via': '', 'search-via': '',
'search-terminatingAt': '', 'search-terminatingAt': '',
'search-headUnit': '', 'select-categoryCode': '',
'search-date-from': '', 'search-date-from': ''
'select-categoryCode': ''
} as Journal.TimetableSearchType); } as Journal.TimetableSearchType);
const countFromIndex = ref(0); const countFromIndex = ref(0);
@@ -278,10 +277,11 @@ export default defineComponent({
this.currentQueryParams['countFrom'] = this.timetableHistory.length; this.currentQueryParams['countFrom'] = this.timetableHistory.length;
const responseData: API.TimetableHistory.Response = await this.apiStore.client.get( const responseData: API.TimetableHistory.Response = await (
'api/getTimetables', await this.apiStore.client!.get('api/getTimetables', {
this.currentQueryParams params: this.currentQueryParams
); })
).data;
if (!responseData) return; if (!responseData) return;
@@ -297,8 +297,6 @@ export default defineComponent({
async fetchHistoryData() { async fetchHistoryData() {
this.extraInfoIndexes.length = 0; this.extraInfoIndexes.length = 0;
const queryParams: API.TimetableHistory.QueryParams = {};
const driverName = this.searchersValues['search-driver'].trim() || undefined; const driverName = this.searchersValues['search-driver'].trim() || undefined;
const trainNo = this.searchersValues['search-train'].trim() || undefined; const trainNo = this.searchersValues['search-train'].trim() || undefined;
const authorName = this.searchersValues['search-dispatcher'].trim() || undefined; const authorName = this.searchersValues['search-dispatcher'].trim() || undefined;
@@ -308,7 +306,6 @@ export default defineComponent({
const via = this.searchersValues['search-via'].trim() || undefined; const via = this.searchersValues['search-via'].trim() || undefined;
const terminatingAt = this.searchersValues['search-terminatingAt'].trim() || undefined; const terminatingAt = this.searchersValues['search-terminatingAt'].trim() || undefined;
const categoryCode = this.searchersValues['select-categoryCode'].trim() || undefined; const categoryCode = this.searchersValues['select-categoryCode'].trim() || undefined;
const headUnit = this.searchersValues['search-headUnit'].trim() || undefined;
let dateFromISO: string | undefined = undefined; let dateFromISO: string | undefined = undefined;
let dateToISO: string | undefined = undefined; let dateToISO: string | undefined = undefined;
@@ -324,6 +321,8 @@ export default defineComponent({
dateToISO = dateTo.toISOString(); dateToISO = dateTo.toISOString();
} }
const queryParams: API.TimetableHistory.QueryParams = {};
this.filterList this.filterList
.filter((f) => f.isActive) .filter((f) => f.isActive)
.forEach((f) => { .forEach((f) => {
@@ -395,27 +394,17 @@ export default defineComponent({
queryParams['sortBy'] = queryParams['sortBy'] =
this.sorterActive.id != 'timetableId' ? this.sorterActive.id : undefined; this.sorterActive.id != 'timetableId' ? this.sorterActive.id : undefined;
// Head unit params
if (headUnit) {
const [headUnitName, headUnitNumber] = headUnit.split('-');
if (headUnitNumber && !isNaN(Number(headUnitNumber))) {
queryParams['headUnitName'] = `${headUnitName}-${headUnitNumber}`;
} else {
queryParams['headUnitType'] = headUnitName;
}
}
if (JSON.stringify(this.currentQueryParams) != JSON.stringify(queryParams)) if (JSON.stringify(this.currentQueryParams) != JSON.stringify(queryParams))
this.dataStatus = Status.Data.Loading; this.dataStatus = Status.Data.Loading;
this.currentQueryParams = queryParams; this.currentQueryParams = queryParams;
try { try {
const responseData: API.TimetableHistory.ResponseShort = await this.apiStore.client.get( const responseData: API.TimetableHistory.ResponseShort = await (
'api/getTimetables', await this.apiStore.client!.get('api/getTimetables', {
this.currentQueryParams params: this.currentQueryParams
); })
).data;
if (!responseData) { if (!responseData) {
this.dataStatus = Status.Data.Error; this.dataStatus = Status.Data.Error;
+24 -20
View File
@@ -40,6 +40,7 @@ import Loading from '../components/Global/Loading.vue';
import ProfileSummary from '../components/PlayerProfileView/ProfileSummary.vue'; import ProfileSummary from '../components/PlayerProfileView/ProfileSummary.vue';
import ProfileRecentStats from '../components/PlayerProfileView/ProfileRecentStats.vue'; import ProfileRecentStats from '../components/PlayerProfileView/ProfileRecentStats.vue';
import ProfileHistoryList from '../components/PlayerProfileView/ProfileHistoryList.vue'; import ProfileHistoryList from '../components/PlayerProfileView/ProfileHistoryList.vue';
import axios from 'axios';
const { t } = useI18n(); const { t } = useI18n();
const router = useRouter(); const router = useRouter();
@@ -70,28 +71,29 @@ onDeactivated(() => {
}); });
async function fetchPlayerInfo(playerId: number) { async function fetchPlayerInfo(playerId: number) {
return apiStore.client.get<API.PlayerInfo.Data>('api/getPlayerInfo', { return apiStore.client!.get<API.PlayerInfo.Data>('api/getPlayerInfo', {
playerId params: {
playerId
}
}); });
} }
async function fetchPlayerJournal(playerId: number) { async function fetchPlayerJournal(playerId: number) {
return apiStore.client.get<API.PlayerJournal.Data>('api/getPlayerJournal', { return apiStore.client!.get<API.PlayerJournal.Data>('api/getPlayerJournal', {
playerId, params: {
dateScope: '30d' playerId,
dateScope: '30d'
}
}); });
} }
async function fetchPlayerTd2Info(playerName: string): Promise<Td2API.UsersInfoByName.Response> { async function fetchPlayerTd2Info(playerName: string) {
const response = await fetch( return axios.get<Td2API.UsersInfoByName.Response>('https://api.td2.info.pl', {
`https://api.td2.info.pl?method=getUsersInfoByName&name=${playerName}` params: {
); method: 'getUsersInfoByName',
name: playerName
if (!response.ok) { }
throw new Error('fetchPlayerTd2Info: could not fetch data'); });
}
return response.json();
} }
async function fetchPlayerData() { async function fetchPlayerData() {
@@ -114,21 +116,23 @@ async function fetchPlayerData() {
const playerInfoResp = await fetchPlayerInfo(playerId.value); const playerInfoResp = await fetchPlayerInfo(playerId.value);
playerName.value = playerName.value =
playerInfoResp.driverStats.driverName || playerInfoResp.dispatcherStats.dispatcherName || ''; playerInfoResp.data.driverStats.driverName ||
playerInfoResp.data.dispatcherStats.dispatcherName ||
'';
if (!playerName.value) { if (!playerName.value) {
router.push('/'); router.push('/');
return; return;
} }
playerInfo.value = playerName.value ? playerInfoResp : undefined; playerInfo.value = playerName.value ? playerInfoResp.data : undefined;
playerInfoStatus.value = Status.Data.Loaded; playerInfoStatus.value = Status.Data.Loaded;
if (playerName.value) { if (playerName.value) {
const playerTD2InfoResp = await fetchPlayerTd2Info(playerName.value); const playerTD2InfoResp = await fetchPlayerTd2Info(playerName.value);
if (playerTD2InfoResp.success && playerTD2InfoResp.message.length == 1) { if (playerTD2InfoResp.data.success && playerTD2InfoResp.data.message.length == 1) {
playerTD2Info.value = playerTD2InfoResp.message[0]; playerTD2Info.value = playerTD2InfoResp.data.message[0];
} }
} }
} catch (error) { } catch (error) {
@@ -140,7 +144,7 @@ async function fetchPlayerData() {
try { try {
const playerJournalResp = await fetchPlayerJournal(playerId.value); const playerJournalResp = await fetchPlayerJournal(playerId.value);
playerJournal.value = playerJournalResp; playerJournal.value = playerJournalResp.data;
playerJournalStatus.value = Status.Data.Loaded; playerJournalStatus.value = Status.Data.Loaded;
} catch (error) { } catch (error) {
playerJournal.value = undefined; playerJournal.value = undefined;
+1 -6
View File
@@ -58,7 +58,6 @@ import SceneryDispatchersHistory from '../components/SceneryView/SceneryDispatch
import { useApiStore } from '../store/apiStore'; import { useApiStore } from '../store/apiStore';
import { Status } from '../typings/common'; import { Status } from '../typings/common';
import SceneryTopList from '../components/SceneryView/SceneryTopList.vue';
const route = useRoute(); const route = useRoute();
const router = useRouter(); const router = useRouter();
@@ -90,10 +89,6 @@ const viewModes = [
{ {
id: 'scenery.option-dispatchers-history', id: 'scenery.option-dispatchers-history',
component: SceneryDispatchersHistory component: SceneryDispatchersHistory
},
{
id: 'scenery.option-top-list',
component: SceneryTopList
} }
]; ];
@@ -189,7 +184,7 @@ function setViewMode(componentName: string) {
background-color: #181818; background-color: #181818;
border-radius: 0.5em; border-radius: 0.5em;
padding: 1em; padding: 1em 0.5em;
} }
.scenery-left { .scenery-left {
+27 -2
View File
@@ -13,6 +13,16 @@
</div> </div>
<div class="topbar-links"> <div class="topbar-links">
<button
v-if="isOldStacjownikDomain"
class="btn--image migrate-info-button"
@click="toggleMigrateInfoCard(true)"
data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t('migrate-info.tooltip-content')}</b>`"
>
<img :src="`/images/icon-alert-triangle.svg`" alt="show migrate info card" />
</button>
<button <button
class="btn--image lang-button" class="btn--image lang-button"
@click="toggleLocales()" @click="toggleLocales()"
@@ -34,7 +44,7 @@
<a <a
class="a-button btn--image gnr-link" class="a-button btn--image gnr-link"
href="https://generator-td2.spythere.eu/" href="https://generator-td2.web.app/"
target="_blank" target="_blank"
data-tooltip-type="HtmlTooltip" data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t('app.gnr-link-content')}</b>`" :data-tooltip-content="`<b>${$t('app.gnr-link-content')}</b>`"
@@ -44,7 +54,7 @@
<a <a
class="a-button btn--image pojazdownik-link" class="a-button btn--image pojazdownik-link"
href="https://pojazdownik-td2.spythere.eu/" href="https://pojazdownik-td2.web.app/"
target="_blank" target="_blank"
data-tooltip-type="HtmlTooltip" data-tooltip-type="HtmlTooltip"
:data-tooltip-content="`<b>${$t('app.pojazdownik-link-content')}</b>`" :data-tooltip-content="`<b>${$t('app.pojazdownik-link-content')}</b>`"
@@ -119,9 +129,19 @@ export default defineComponent({
this.isDonationCardOpen = value; this.isDonationCardOpen = value;
}, },
toggleMigrateInfoCard(value: boolean) {
this.mainStore.isMigrateInfoCardOpen = value;
},
toggleLocales() { toggleLocales() {
this.mainStore.changeLocale(this.mainStore.currentLocale == 'pl' ? 'en' : 'pl'); this.mainStore.changeLocale(this.mainStore.currentLocale == 'pl' ? 'en' : 'pl');
} }
},
computed: {
isOldStacjownikDomain() {
return location.hostname == 'stacjownik-td2.web.app';
}
} }
}); });
</script> </script>
@@ -180,6 +200,11 @@ button.lang-button {
background-color: #111; background-color: #111;
} }
button.migrate-info-button {
padding: 0 0.5em;
background-color: var(--clr-primary);
}
a.pojazdownik-link { a.pojazdownik-link {
background-color: #1f263b; background-color: #1f263b;
+5 -1
View File
@@ -116,10 +116,13 @@ export default defineComponent({
<style lang="scss" scoped> <style lang="scss" scoped>
@use '../styles/responsive'; @use '../styles/responsive';
.trains-view {
position: relative;
}
.trains_wrapper { .trains_wrapper {
margin: 1rem auto; margin: 1rem auto;
max-width: var(--max-container-width); max-width: var(--max-container-width);
position: relative;
} }
.trains_topbar { .trains_topbar {
@@ -127,6 +130,7 @@ export default defineComponent({
align-items: center; align-items: center;
gap: 0.5em; gap: 0.5em;
position: relative;
margin-bottom: 0.5em; margin-bottom: 0.5em;
} }
-15
View File
@@ -1,15 +0,0 @@
{
"extends": "@vue/tsconfig/tsconfig.dom.json",
"include": ["env.d.ts", "src/**/*", "src/**/*.vue"],
"exclude": ["src/**/__tests__/*"],
"compilerOptions": {
"noUncheckedIndexedAccess": false,
"verbatimModuleSyntax": null,
"paths": {
"@/*": ["./src/*"]
},
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.app.tsbuildinfo"
}
}
+17 -4
View File
@@ -1,11 +1,24 @@
{ {
"files": [], "compilerOptions": {
"target": "ESNext",
"useDefineForClassFields": true,
"module": "ESNext",
"moduleResolution": "Node",
"strict": true,
"jsx": "preserve",
"sourceMap": true,
"resolveJsonModule": true,
"isolatedModules": true,
"esModuleInterop": true,
"lib": ["ESNext", "DOM"],
"types": ["vite/client", "vite-plugin-pwa/client"],
"skipLibCheck": true
},
"include": ["src/**/*.ts", "src/**/*.d.ts", "src/**/*.tsx", "src/**/*.vue"],
"references": [ "references": [
{ {
"path": "./tsconfig.node.json" "path": "./tsconfig.node.json"
},
{
"path": "./tsconfig.app.json"
} }
] ]
} }
+6 -9
View File
@@ -1,12 +1,9 @@
// TSConfig for modules that run in Node.js environment via either transpilation or type-stripping.
{ {
"extends": "@tsconfig/node24/tsconfig.json",
"include": ["vite.config.*", "eslint.config.*"],
"compilerOptions": { "compilerOptions": {
"module": "preserve", "composite": true,
"moduleResolution": "bundler", "module": "nodenext",
"types": ["node", "vite/client", "vite-plugin-pwa/client"], "moduleResolution": "nodenext",
"noEmit": true, "allowSyntheticDefaultImports": true
"tsBuildInfoFile": "./node_modules/.tmp/tsconfig.node.tsbuildinfo" },
} "include": ["vite.config.ts"]
} }
+4 -7
View File
@@ -1,7 +1,7 @@
import { defineConfig } from 'vite'; import { defineConfig } from 'vite';
import vue from '@vitejs/plugin-vue'; import vue from '@vitejs/plugin-vue';
import { VitePWA } from 'vite-plugin-pwa'; import { VitePWA } from 'vite-plugin-pwa';
import { fileURLToPath } from 'node:url'; import path from 'path';
export default defineConfig({ export default defineConfig({
server: { port: 5123, open: false }, server: { port: 5123, open: false },
@@ -14,7 +14,7 @@ export default defineConfig({
}, },
resolve: { resolve: {
alias: { alias: {
'@': fileURLToPath(new URL('./src', import.meta.url)) '@': path.resolve(__dirname, 'src')
} }
}, },
plugins: [ plugins: [
@@ -29,13 +29,10 @@ export default defineConfig({
{ {
urlPattern: urlPattern:
/^https:\/\/stacjownik.spythere.eu\/api\/(getVehiclesData|getDonators|getSceneries)/i, /^https:\/\/stacjownik.spythere.eu\/api\/(getVehiclesData|getDonators|getSceneries)/i,
handler: 'StaleWhileRevalidate', handler: 'NetworkFirst',
options: { options: {
cacheName: 'stacjownik-api-cache', cacheName: 'stacjownik-api-cache',
cacheableResponse: { statuses: [0, 200] }, cacheableResponse: { statuses: [0, 200] }
expiration: {
maxAgeSeconds: 3600
}
} }
} }
] ]
+1500 -1501
View File
File diff suppressed because it is too large Load Diff