mirror of
https://github.com/Spythere/stacjownik.git
synced 2026-05-03 05:18:11 +00:00
@@ -3,10 +3,10 @@ name: Build & Deploy to VPS
|
|||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- master
|
- main
|
||||||
|
|
||||||
env:
|
env:
|
||||||
PROJECT_NAME: stacjownik
|
PROJECT_NAME: stacjownik-td2
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build_and_deploy:
|
build_and_deploy:
|
||||||
|
|||||||
+2
-2
@@ -15,8 +15,8 @@ app.get('/api/getSceneries', (_, res) => {
|
|||||||
res.sendFile(path.join(cwd(), 'endpoints', 'getSceneries.json'));
|
res.sendFile(path.join(cwd(), 'endpoints', 'getSceneries.json'));
|
||||||
});
|
});
|
||||||
|
|
||||||
app.get('/api/getVehicles', (_, res) => {
|
app.get('/api/getVehiclesData', (_, res) => {
|
||||||
res.sendFile(path.join(cwd(), 'endpoints', 'getVehicles.json'));
|
res.sendFile(path.join(cwd(), 'endpoints', 'getVehiclesData.json'));
|
||||||
});
|
});
|
||||||
|
|
||||||
app.get('/api/getDonators', (_, res) => {
|
app.get('/api/getDonators', (_, res) => {
|
||||||
|
|||||||
+1
-1
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "stacjownik",
|
"name": "stacjownik",
|
||||||
"version": "1.30.7",
|
"version": "1.31.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<Card :is-open="isUpdateCardOpen" @toggle-card="toggleCard(false)">
|
<Card :is-open="isUpdateCardOpen" @toggle-card="toggleCard(false)">
|
||||||
<div class="content">
|
<div class="content" tabindex="0" ref="content">
|
||||||
<h1 style="margin-bottom: 0.5em">🚀 {{ $t('update.title') }}</h1>
|
<h1 style="margin-bottom: 0.5em">🚀 {{ $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>
|
||||||
@@ -13,7 +13,14 @@
|
|||||||
<p class="bottom-info">
|
<p class="bottom-info">
|
||||||
{{ $t('update.info-1') }}
|
{{ $t('update.info-1') }}
|
||||||
<br />
|
<br />
|
||||||
<span v-html="$t('update.info-2')"></span>
|
|
||||||
|
<i18n-t keypath="update.info-2">
|
||||||
|
<template v-slot:link>
|
||||||
|
<a href="https://github.com/Spythere/stacjownik" target="_blank">{{
|
||||||
|
$t('update.info-2-link-text')
|
||||||
|
}}</a>
|
||||||
|
</template>
|
||||||
|
</i18n-t>
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</Card>
|
</Card>
|
||||||
@@ -51,7 +58,7 @@ export default defineComponent({
|
|||||||
watch: {
|
watch: {
|
||||||
isUpdateCardOpen(val: boolean) {
|
isUpdateCardOpen(val: boolean) {
|
||||||
this.$nextTick(() => {
|
this.$nextTick(() => {
|
||||||
if (val) (this.$refs['confirm-btn'] as HTMLElement).focus();
|
if (val) (this.$refs['content'] as HTMLElement).focus();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -79,13 +86,18 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
|
|
||||||
::v-deep(h2) {
|
::v-deep(h2) {
|
||||||
padding: 0.25em 0;
|
margin-top: 1em;
|
||||||
|
padding: 0.5em 0;
|
||||||
border-bottom: 1px solid #aaa;
|
border-bottom: 1px solid #aaa;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
::v-deep(h3) {
|
||||||
|
padding: 0.5em 0;
|
||||||
|
}
|
||||||
|
|
||||||
::v-deep(ul) {
|
::v-deep(ul) {
|
||||||
list-style: initial;
|
list-style: disc;
|
||||||
padding: 1em;
|
padding: 0 1.5em;
|
||||||
line-height: 1.5em;
|
line-height: 1.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -105,7 +117,7 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
|
|
||||||
button {
|
button {
|
||||||
margin: 0 auto;
|
margin: 0.5em auto;
|
||||||
padding: 0.5em 0.75em;
|
padding: 0.5em 0.75em;
|
||||||
font-size: 1.1em;
|
font-size: 1.1em;
|
||||||
}
|
}
|
||||||
@@ -117,5 +129,6 @@ p.bottom-info {
|
|||||||
|
|
||||||
a {
|
a {
|
||||||
text-decoration: underline;
|
text-decoration: underline;
|
||||||
|
color: white;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -205,7 +205,7 @@ const availableCategories = computed(() => {
|
|||||||
for (const stockName of stockList) {
|
for (const stockName of stockList) {
|
||||||
const [vehicleName, ...cargoList] = stockName.split(':');
|
const [vehicleName, ...cargoList] = stockName.split(':');
|
||||||
|
|
||||||
const vehicleData = apiStore.vehiclesData?.find((v) => v.name == vehicleName);
|
const vehicleData = apiStore.vehiclesData?.vehicles.find((v) => v.name == vehicleName);
|
||||||
|
|
||||||
if (!vehicleData) continue;
|
if (!vehicleData) continue;
|
||||||
|
|
||||||
|
|||||||
@@ -1,23 +0,0 @@
|
|||||||
<template>
|
|
||||||
<button class="action-btn btn--filled">
|
|
||||||
<div class="button_content">
|
|
||||||
<slot></slot>
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script lang="ts">
|
|
||||||
import { defineComponent } from 'vue';
|
|
||||||
|
|
||||||
export default defineComponent({});
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style lang="scss">
|
|
||||||
@use '../../styles/responsive';
|
|
||||||
|
|
||||||
.button_content {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -10,6 +10,7 @@ export namespace Journal {
|
|||||||
| 'search-train'
|
| 'search-train'
|
||||||
| 'search-date-from'
|
| 'search-date-from'
|
||||||
| 'search-dispatcher'
|
| 'search-dispatcher'
|
||||||
|
| 'search-includesScenery'
|
||||||
| 'search-issuedFrom'
|
| 'search-issuedFrom'
|
||||||
| 'search-terminatingAt'
|
| 'search-terminatingAt'
|
||||||
| 'search-via'
|
| 'search-via'
|
||||||
|
|||||||
@@ -151,6 +151,7 @@ export default defineComponent({
|
|||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@use '../../styles/responsive';
|
@use '../../styles/responsive';
|
||||||
@use '../../styles/scenery-history-table';
|
@use '../../styles/scenery-history-table';
|
||||||
|
@use '../../styles/badge';
|
||||||
|
|
||||||
.scenery-dispatchers-history {
|
.scenery-dispatchers-history {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
|
|||||||
@@ -57,20 +57,8 @@ export default defineComponent({
|
|||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss" scoped>
|
||||||
@use '../../styles/responsive';
|
@use '../../styles/responsive';
|
||||||
@use '../../styles/badge';
|
|
||||||
|
|
||||||
h3.section-header {
|
|
||||||
margin: 0.5em 0;
|
|
||||||
padding: 0.3em;
|
|
||||||
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
font-size: 1.2em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.info-lists {
|
.info-lists {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<section class="info-spawn-list">
|
<section class="info-spawn-list">
|
||||||
<h3 class="spawn-header section-header">
|
<h3 class="spawn-header">
|
||||||
<img src="/images/icon-spawn.svg" alt="Open spawns icon" />
|
<img src="/images/icon-spawn.svg" alt="Open spawns icon" />
|
||||||
{{ $t('scenery.spawns') }}
|
{{ $t('scenery.spawns') }}
|
||||||
<span class="text--primary">{{ onlineScenery?.spawns.length || '0' }}</span>
|
<span class="text--primary">{{ onlineScenery?.spawns.length || '0' }}</span>
|
||||||
@@ -53,10 +53,23 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
|
@use '../../../styles/badge';
|
||||||
|
|
||||||
ul {
|
ul {
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
h3.spawn-header {
|
||||||
|
margin: 0.5em 0;
|
||||||
|
padding: 0.3em;
|
||||||
|
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
font-size: 1.2em;
|
||||||
|
}
|
||||||
|
|
||||||
.spawns-anim {
|
.spawns-anim {
|
||||||
&-move,
|
&-move,
|
||||||
&-enter-active,
|
&-enter-active,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<section class="info-user-list">
|
<section class="info-user-list">
|
||||||
<h3 class="user-header section-header">
|
<h3 class="user-header">
|
||||||
<img src="/images/icon-user.svg" alt="Users icon" />
|
<img src="/images/icon-user.svg" alt="Users icon" />
|
||||||
{{ $t('scenery.users') }}
|
{{ $t('scenery.users') }}
|
||||||
<span class="text--primary">{{ onlineScenery?.stationTrains?.length || 0 }}</span
|
<span class="text--primary">{{ onlineScenery?.stationTrains?.length || 0 }}</span
|
||||||
@@ -111,6 +111,8 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
|
@use '../../../styles/badge';
|
||||||
|
|
||||||
$no-timetable: #aaa;
|
$no-timetable: #aaa;
|
||||||
$departed: springgreen;
|
$departed: springgreen;
|
||||||
$stopped: #ffa600;
|
$stopped: #ffa600;
|
||||||
@@ -118,6 +120,17 @@ $online: gold;
|
|||||||
$terminated: salmon;
|
$terminated: salmon;
|
||||||
$disconnected: slategray;
|
$disconnected: slategray;
|
||||||
|
|
||||||
|
h3.user-header {
|
||||||
|
margin: 0.5em 0;
|
||||||
|
padding: 0.3em;
|
||||||
|
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
font-size: 1.2em;
|
||||||
|
}
|
||||||
|
|
||||||
.info-user-list {
|
.info-user-list {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,6 +35,7 @@
|
|||||||
id="scenery-search"
|
id="scenery-search"
|
||||||
list="sceneries"
|
list="sceneries"
|
||||||
:placeholder="$t('filters.sceneries-placeholder')"
|
:placeholder="$t('filters.sceneries-placeholder')"
|
||||||
|
@change="handleSceneriesInput"
|
||||||
@focus="preventKeyDown = true"
|
@focus="preventKeyDown = true"
|
||||||
@blur="preventKeyDown = false"
|
@blur="preventKeyDown = false"
|
||||||
/>
|
/>
|
||||||
@@ -44,42 +45,40 @@
|
|||||||
</button>
|
</button>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section class="card_input-search authors">
|
<section class="card_input-search">
|
||||||
<datalist id="authors" name="authors">
|
|
||||||
<option v-for="(author, i) in authorsOptions" :key="i" :value="author"></option>
|
|
||||||
</datalist>
|
|
||||||
|
|
||||||
<input
|
<input
|
||||||
type="text"
|
v-model="filters['lines']"
|
||||||
id="author"
|
id="line-numbers-search"
|
||||||
list="authors"
|
:placeholder="$t('filters.line-numbers-placeholder')"
|
||||||
name="authors"
|
|
||||||
v-model="filters['authors']"
|
|
||||||
:placeholder="$t('filters.authors-placeholder')"
|
|
||||||
@focus="preventKeyDown = true"
|
@focus="preventKeyDown = true"
|
||||||
@blur="preventKeyDown = false"
|
@blur="preventKeyDown = false"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<button class="btn--action btn--image" @click="resetLineNumbersInput">
|
||||||
|
<img src="/images/icon-exit.svg" alt="reset line numbers search" />
|
||||||
|
</button>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="card_input-search">
|
||||||
|
<select id="author" name="authors" v-model="filters['authors']">
|
||||||
|
<option value="">{{ $t('filters.authors-placeholder') }}</option>
|
||||||
|
<option v-for="(author, i) in authorsOptions" :key="i" :value="author">
|
||||||
|
{{ author }}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
|
||||||
<button class="btn--action btn--image" @click="resetAuthorsInput">
|
<button class="btn--action btn--image" @click="resetAuthorsInput">
|
||||||
<img src="/images/icon-exit.svg" alt="reset authors search" />
|
<img src="/images/icon-exit.svg" alt="reset authors search" />
|
||||||
</button>
|
</button>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section class="card_input-search">
|
<section class="card_input-search">
|
||||||
<datalist id="projects" name="projects">
|
<select id="projects" name="projects" v-model="filters['projects']">
|
||||||
<option v-for="(project, i) in projectsOptions" :key="i" :value="project"></option>
|
<option value="">{{ $t('filters.projects-placeholder') }}</option>
|
||||||
</datalist>
|
<option v-for="(project, i) in projectsOptions" :key="i" :value="project">
|
||||||
|
{{ project }}
|
||||||
<input
|
</option>
|
||||||
type="text"
|
</select>
|
||||||
id="projects"
|
|
||||||
list="projects"
|
|
||||||
name="projects"
|
|
||||||
v-model="filters['projects']"
|
|
||||||
:placeholder="$t('filters.projects-placeholder')"
|
|
||||||
@focus="preventKeyDown = true"
|
|
||||||
@blur="preventKeyDown = false"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<button class="btn--action btn--image" @click="resetProjectsInput">
|
<button class="btn--action btn--image" @click="resetProjectsInput">
|
||||||
<img src="/images/icon-exit.svg" alt="reset projects search" />
|
<img src="/images/icon-exit.svg" alt="reset projects search" />
|
||||||
@@ -92,7 +91,7 @@
|
|||||||
v-for="(sectionFilters, sectionKey) in filtersSections"
|
v-for="(sectionFilters, sectionKey) in filtersSections"
|
||||||
:key="sectionKey"
|
:key="sectionKey"
|
||||||
>
|
>
|
||||||
<h3 class="text--primary">
|
<h3 class="section-header">
|
||||||
<span class="active-indicator" v-if="!areSectionFiltersDefault(sectionKey)"></span>
|
<span class="active-indicator" v-if="!areSectionFiltersDefault(sectionKey)"></span>
|
||||||
{{ $t(`filters.sections.${sectionKey}`) }}
|
{{ $t(`filters.sections.${sectionKey}`) }}
|
||||||
<button @click="resetSectionFilters(sectionKey)">RESET</button>
|
<button @click="resetSectionFilters(sectionKey)">RESET</button>
|
||||||
@@ -122,7 +121,7 @@
|
|||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section class="card_timestamp">
|
<section class="card_timestamp">
|
||||||
<h3 class="section-header">{{ $t('filters.minimum-hours-title') }}</h3>
|
<h3 class="hours-section-header">{{ $t('filters.minimum-hours-title') }}</h3>
|
||||||
|
|
||||||
<span class="clock">
|
<span class="clock">
|
||||||
<button class="btn--action" @click="subHour">-</button>
|
<button class="btn--action" @click="subHour">-</button>
|
||||||
@@ -217,8 +216,6 @@ export default defineComponent({
|
|||||||
sliderStates,
|
sliderStates,
|
||||||
|
|
||||||
minimumHours: 0,
|
minimumHours: 0,
|
||||||
authorSearchFilter: '',
|
|
||||||
projectSearchFilter: '',
|
|
||||||
|
|
||||||
currentRegion: { id: '', value: '' },
|
currentRegion: { id: '', value: '' },
|
||||||
|
|
||||||
@@ -276,6 +273,8 @@ export default defineComponent({
|
|||||||
authorsOptions() {
|
authorsOptions() {
|
||||||
return this.store.stationList
|
return this.store.stationList
|
||||||
.reduce((acc, station) => {
|
.reduce((acc, station) => {
|
||||||
|
if (station.generalInfo?.hidden === true) return acc;
|
||||||
|
|
||||||
station.generalInfo?.authors?.forEach((author) => {
|
station.generalInfo?.authors?.forEach((author) => {
|
||||||
if (author.trim() != '' && !acc.includes(author.toLocaleLowerCase()))
|
if (author.trim() != '' && !acc.includes(author.toLocaleLowerCase()))
|
||||||
acc.push(author.toLocaleLowerCase());
|
acc.push(author.toLocaleLowerCase());
|
||||||
@@ -289,8 +288,10 @@ export default defineComponent({
|
|||||||
projectsOptions() {
|
projectsOptions() {
|
||||||
return this.store.stationList
|
return this.store.stationList
|
||||||
.reduce((acc, station) => {
|
.reduce((acc, station) => {
|
||||||
if (!station.generalInfo || !station.generalInfo.project || station.generalInfo.hidden) return acc;
|
if (!station.generalInfo || !station.generalInfo.project || station.generalInfo.hidden)
|
||||||
if (!acc.includes(station.generalInfo.project.trim())) acc.push(station.generalInfo.project.trim());
|
return acc;
|
||||||
|
if (!acc.includes(station.generalInfo.project.trim()))
|
||||||
|
acc.push(station.generalInfo.project.trim());
|
||||||
|
|
||||||
return acc;
|
return acc;
|
||||||
}, [] as string[])
|
}, [] as string[])
|
||||||
@@ -320,11 +321,15 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
resetAuthorsInput() {
|
resetAuthorsInput() {
|
||||||
this.filters['authors'] = this.authorSearchFilter;
|
this.filters['authors'] = '';
|
||||||
},
|
},
|
||||||
|
|
||||||
resetProjectsInput() {
|
resetProjectsInput() {
|
||||||
this.filters['projects'] = this.projectSearchFilter;
|
this.filters['projects'] = '';
|
||||||
|
},
|
||||||
|
|
||||||
|
resetLineNumbersInput() {
|
||||||
|
this.filters['lines'] = '';
|
||||||
},
|
},
|
||||||
|
|
||||||
handleSceneriesInput() {
|
handleSceneriesInput() {
|
||||||
@@ -369,7 +374,6 @@ export default defineComponent({
|
|||||||
|
|
||||||
// Reset local model values
|
// Reset local model values
|
||||||
this.minimumHours = 0;
|
this.minimumHours = 0;
|
||||||
this.authorSearchFilter = '';
|
|
||||||
|
|
||||||
// Reset global filters
|
// Reset global filters
|
||||||
Object.keys(this.filters).forEach((filterKey) => {
|
Object.keys(this.filters).forEach((filterKey) => {
|
||||||
@@ -413,6 +417,14 @@ export default defineComponent({
|
|||||||
@use '../../styles/animations';
|
@use '../../styles/animations';
|
||||||
|
|
||||||
h3.section-header {
|
h3.section-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 0.25em;
|
||||||
|
gap: 0.5em;
|
||||||
|
color: var(--clr-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
h3.hours-section-header {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
margin: 0.5em 0;
|
margin: 0.5em 0;
|
||||||
}
|
}
|
||||||
@@ -494,15 +506,12 @@ h3.section-header {
|
|||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
input {
|
input,
|
||||||
|
select {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 0.5em;
|
padding: 0.5em;
|
||||||
border: 1px solid #aaa;
|
border: 1px solid #aaa;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.authors {
|
|
||||||
margin-top: 1em;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.section-filters {
|
.section-filters {
|
||||||
@@ -573,12 +582,6 @@ h3.section-header {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.option-section h3 {
|
.option-section h3 {
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
margin-bottom: 0.25em;
|
|
||||||
|
|
||||||
gap: 0.5em;
|
|
||||||
|
|
||||||
button {
|
button {
|
||||||
padding: 0.15em;
|
padding: 0.15em;
|
||||||
color: coral;
|
color: coral;
|
||||||
|
|||||||
@@ -14,7 +14,7 @@
|
|||||||
class="header-text"
|
class="header-text"
|
||||||
:class="headerName"
|
:class="headerName"
|
||||||
>
|
>
|
||||||
<span class="header_wrapper">
|
<div class="header_wrapper">
|
||||||
<div v-html="$t(`sceneries.headers.${headerName}`)"></div>
|
<div v-html="$t(`sceneries.headers.${headerName}`)"></div>
|
||||||
|
|
||||||
<img
|
<img
|
||||||
@@ -23,7 +23,7 @@
|
|||||||
:src="`/images/icon-arrow-${activeSorter.dir == 1 ? 'asc' : 'desc'}.svg`"
|
:src="`/images/icon-arrow-${activeSorter.dir == 1 ? 'asc' : 'desc'}.svg`"
|
||||||
alt="sort icon"
|
alt="sort icon"
|
||||||
/>
|
/>
|
||||||
</span>
|
</div>
|
||||||
</th>
|
</th>
|
||||||
|
|
||||||
<th
|
<th
|
||||||
@@ -52,14 +52,14 @@
|
|||||||
</thead>
|
</thead>
|
||||||
|
|
||||||
<tbody>
|
<tbody>
|
||||||
<router-link
|
<tr
|
||||||
v-for="station in filteredStationList"
|
v-for="station in filteredStationList"
|
||||||
class="a-row"
|
class="a-row"
|
||||||
role="row"
|
tabindex="0"
|
||||||
:key="station.name"
|
:key="station.name"
|
||||||
@click.right.prevent="openForumSite($event, station.generalInfo?.url)"
|
@click.right.prevent="openForumSite($event, station.generalInfo?.url)"
|
||||||
@keydown.space.prevent="openForumSite($event, station.generalInfo?.url)"
|
@click="getSceneryRoute(station)"
|
||||||
:to="getSceneryRoute(station)"
|
@keydown.enter="getSceneryRoute(station)"
|
||||||
>
|
>
|
||||||
<td class="station-name" :class="station.generalInfo?.availability">
|
<td class="station-name" :class="station.generalInfo?.availability">
|
||||||
<b v-if="station.generalInfo?.project" style="color: salmon">{{
|
<b v-if="station.generalInfo?.project" style="color: salmon">{{
|
||||||
@@ -314,7 +314,7 @@
|
|||||||
>
|
>
|
||||||
{{ station.onlineInfo?.scheduledTrainCount.confirmed ?? '-' }}
|
{{ station.onlineInfo?.scheduledTrainCount.confirmed ?? '-' }}
|
||||||
</td>
|
</td>
|
||||||
</router-link>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
@@ -384,15 +384,13 @@ export default defineComponent({
|
|||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
getSceneryRoute(station: Station) {
|
getSceneryRoute(station: Station) {
|
||||||
// TODO: Hide tooltips when navigating away
|
this.$router.push({
|
||||||
|
|
||||||
return {
|
|
||||||
name: 'SceneryView',
|
name: 'SceneryView',
|
||||||
query: {
|
query: {
|
||||||
station: station.name,
|
station: station.name,
|
||||||
region: this.$route.query.region || undefined
|
region: this.$route.query.region || undefined
|
||||||
}
|
}
|
||||||
};
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
openDonationCard(e: Event) {
|
openDonationCard(e: Event) {
|
||||||
@@ -459,78 +457,78 @@ table {
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
min-width: 1250px;
|
min-width: 1250px;
|
||||||
white-space: wrap;
|
white-space: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
thead {
|
thead {
|
||||||
position: sticky;
|
position: sticky;
|
||||||
top: 0;
|
top: 0;
|
||||||
|
z-index: 50;
|
||||||
|
}
|
||||||
|
|
||||||
|
thead tr {
|
||||||
|
background-color: var(--clr-bg3);
|
||||||
|
}
|
||||||
|
|
||||||
|
thead th {
|
||||||
|
background-color: var(--clr-bg3);
|
||||||
|
white-space: pre-wrap;
|
||||||
|
padding: 0.5em 0.25em;
|
||||||
|
|
||||||
|
cursor: pointer;
|
||||||
|
user-select: none;
|
||||||
|
-moz-user-select: none;
|
||||||
|
-webkit-user-select: none;
|
||||||
|
|
||||||
|
&.station {
|
||||||
|
width: 12em;
|
||||||
}
|
}
|
||||||
|
|
||||||
thead tr {
|
&.min-lvl {
|
||||||
background-color: var(--clr-bg3);
|
width: 4em;
|
||||||
}
|
}
|
||||||
|
|
||||||
thead th {
|
&.status {
|
||||||
&.station {
|
width: 10em;
|
||||||
width: 12em;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
&.min-lvl {
|
&.dispatcher {
|
||||||
width: 4em;
|
width: 12em;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.status {
|
&.dispatcher-lvl {
|
||||||
width: 10em;
|
width: 6em;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.dispatcher {
|
&.routes-double,
|
||||||
width: 12em;
|
&.routes-single {
|
||||||
}
|
width: 7em;
|
||||||
|
}
|
||||||
|
|
||||||
&.dispatcher-lvl {
|
&.general {
|
||||||
width: 6em;
|
width: 11em;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.routes-double,
|
&.header-image {
|
||||||
&.routes-single {
|
width: 3.5em;
|
||||||
width: 7em;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.general {
|
&.user {
|
||||||
width: 11em;
|
width: 5em;
|
||||||
}
|
|
||||||
|
|
||||||
&.header-image {
|
|
||||||
width: 3.5em;
|
|
||||||
|
|
||||||
&.user {
|
|
||||||
width: 5em;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
padding: 0.5em 0.25em;
|
|
||||||
background-color: var(--clr-bg3);
|
|
||||||
white-space: pre-wrap;
|
|
||||||
|
|
||||||
cursor: pointer;
|
|
||||||
user-select: none;
|
|
||||||
-moz-user-select: none;
|
|
||||||
-webkit-user-select: none;
|
|
||||||
|
|
||||||
span {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
|
|
||||||
img {
|
|
||||||
width: 1.5em;
|
|
||||||
vertical-align: middle;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
tr,
|
thead th .header_wrapper {
|
||||||
.a-row {
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
|
img {
|
||||||
|
width: 1.5em;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tbody tr {
|
||||||
background-color: $rowCol;
|
background-color: $rowCol;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
|
|
||||||
@@ -550,6 +548,7 @@ tr,
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
|
height: 2.5em;
|
||||||
|
|
||||||
&.inactive {
|
&.inactive {
|
||||||
opacity: 0.2;
|
opacity: 0.2;
|
||||||
|
|||||||
@@ -145,13 +145,33 @@ function filterSliderValues(filters: Record<string, any>, generalInfo: StationGe
|
|||||||
}
|
}
|
||||||
|
|
||||||
function filterInputValues(filters: Record<string, any>, generalInfo: StationGeneralInfo) {
|
function filterInputValues(filters: Record<string, any>, generalInfo: StationGeneralInfo) {
|
||||||
return (
|
if (
|
||||||
(filters['authors'].length > 3 &&
|
filters['authors'].length > 3 &&
|
||||||
!generalInfo.authors
|
generalInfo.authors &&
|
||||||
?.map((a) => a.toLocaleLowerCase())
|
!generalInfo.authors.some(
|
||||||
.includes(filters['authors'].toLocaleLowerCase())) ||
|
(a) => a.toLocaleLowerCase() == filters['authors'].toLocaleLowerCase()
|
||||||
(filters['projects'].length > 0 && generalInfo.project != filters['projects'])
|
)
|
||||||
);
|
)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
if (filters['projects'].length > 0 && generalInfo.project != filters['projects']) return true;
|
||||||
|
|
||||||
|
if (filters['lines'].length > 0) {
|
||||||
|
const linesNumbers = (filters['lines'] as string)
|
||||||
|
.split(',')
|
||||||
|
.map((l) => Number(l))
|
||||||
|
.filter((l) => !isNaN(l) && l != 0);
|
||||||
|
|
||||||
|
if (
|
||||||
|
!generalInfo.lines
|
||||||
|
?.split(',')
|
||||||
|
.map((l) => Number(l))
|
||||||
|
.some((l) => linesNumbers.includes(l))
|
||||||
|
)
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const sortStations = (a: Station, b: Station, sorter: ActiveSorter) => {
|
export const sortStations = (a: Station, b: Station, sorter: ActiveSorter) => {
|
||||||
|
|||||||
@@ -18,9 +18,9 @@
|
|||||||
<span v-if="vehicleCargo">({{ vehicleCargo.id }})</span>
|
<span v-if="vehicleCargo">({{ vehicleCargo.id }})</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="vehicle-props" v-if="vehicleData">
|
<div class="vehicle-props" v-if="vehicleGroup">
|
||||||
{{ vehicleData.group.speed }}km/h • {{ vehicleData.group.length }}m •
|
{{ vehicleGroup.speed }}km/h • {{ vehicleGroup.length }}m •
|
||||||
{{ (vehicleData.group.weight / 1000).toFixed(1) }}t
|
{{ (vehicleGroup.weight / 1000).toFixed(1) }}t
|
||||||
<span v-if="vehicleCargo">(+{{ (vehicleCargo.weight / 1000).toFixed(1) }}t)</span>
|
<span v-if="vehicleCargo">(+{{ (vehicleCargo.weight / 1000).toFixed(1) }}t)</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -73,12 +73,18 @@ export default defineComponent({
|
|||||||
return this.tooltipStore.content.split(':')[0];
|
return this.tooltipStore.content.split(':')[0];
|
||||||
},
|
},
|
||||||
|
|
||||||
vehicleData() {
|
vehicleGroup() {
|
||||||
return this.apiStore.vehiclesData?.find((v) => v.name == this.vehicleName);
|
if (!this.apiStore.vehiclesData) return null;
|
||||||
|
|
||||||
|
const vehicle = this.apiStore.vehiclesData.vehicles.find((v) => v.name == this.vehicleName);
|
||||||
|
|
||||||
|
if (!vehicle) return null;
|
||||||
|
|
||||||
|
return this.apiStore.vehiclesData.vehicleGroups.find((g) => g.id == vehicle?.vehicleGroupsId);
|
||||||
},
|
},
|
||||||
|
|
||||||
vehicleCargo() {
|
vehicleCargo() {
|
||||||
const x = this.vehicleData?.group.cargoTypes?.find(
|
const x = this.vehicleGroup?.cargoTypes?.find(
|
||||||
(c) => c.id == this.tooltipStore.content.split(':')[1]
|
(c) => c.id == this.tooltipStore.content.split(':')[1]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
+6
-3
@@ -69,7 +69,8 @@
|
|||||||
"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",
|
||||||
"info-2": "The full app changelog available on <a href='https://github.com/Spythere/stacjownik' target='_blank'>the project's GitHub</a>"
|
"info-2": "The full app changelog available on {link}",
|
||||||
|
"info-2-link-text": "the project's GitHub page"
|
||||||
},
|
},
|
||||||
"app": {
|
"app": {
|
||||||
"sceneries": "SCENERIES",
|
"sceneries": "SCENERIES",
|
||||||
@@ -199,6 +200,7 @@
|
|||||||
"search-dispatcher": "Dispatcher name",
|
"search-dispatcher": "Dispatcher name",
|
||||||
"search-station": "Scenery name / #",
|
"search-station": "Scenery name / #",
|
||||||
"search-author": "Timetable author name",
|
"search-author": "Timetable author name",
|
||||||
|
"search-includesScenery": "Includes scenery name",
|
||||||
"search-issuedFrom": "Issuing scenery name",
|
"search-issuedFrom": "Issuing scenery name",
|
||||||
"search-via": "Via scenery name",
|
"search-via": "Via scenery name",
|
||||||
"search-terminatingAt": "Terminating scenery name",
|
"search-terminatingAt": "Terminating scenery name",
|
||||||
@@ -316,8 +318,9 @@
|
|||||||
"minTwoWayInt": "MIN. INTERNAL OTHER DOUBLE TRACK ROUTES"
|
"minTwoWayInt": "MIN. INTERNAL OTHER DOUBLE TRACK ROUTES"
|
||||||
},
|
},
|
||||||
"sceneries-placeholder": "Search for scenery",
|
"sceneries-placeholder": "Search for scenery",
|
||||||
"authors-placeholder": "Scenery author (other filters apply)",
|
"line-numbers-placeholder": "Line numbers (separated by commas)",
|
||||||
"projects-placeholder": "Scenery project (other filters apply)",
|
"authors-placeholder": "Scenery author",
|
||||||
|
"projects-placeholder": "Scenery project",
|
||||||
"search-button-title": "SEARCH",
|
"search-button-title": "SEARCH",
|
||||||
"minimum-hours-title": "SHOW ONLY SCENERIES UNTIL:",
|
"minimum-hours-title": "SHOW ONLY SCENERIES UNTIL:",
|
||||||
"now": "NOW",
|
"now": "NOW",
|
||||||
|
|||||||
+6
-3
@@ -68,7 +68,8 @@
|
|||||||
"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",
|
||||||
"info-2": "Pełny changelog dostępny na <a href='https://github.com/Spythere/stacjownik' target='_blank'>GitHubie projektu</a>"
|
"info-2": "Pełny changelog dostępny na {link}",
|
||||||
|
"info-2-link-text": "GitHubie projektu"
|
||||||
},
|
},
|
||||||
"app": {
|
"app": {
|
||||||
"sceneries": "SCENERIE",
|
"sceneries": "SCENERIE",
|
||||||
@@ -195,6 +196,7 @@
|
|||||||
"search-dispatcher": "Nick dyżurnego",
|
"search-dispatcher": "Nick dyżurnego",
|
||||||
"search-station": "Nazwa scenerii / #",
|
"search-station": "Nazwa scenerii / #",
|
||||||
"search-author": "Nick autora rozkładu jazdy",
|
"search-author": "Nick autora rozkładu jazdy",
|
||||||
|
"search-includesScenery": "Zawiera scenerię",
|
||||||
"search-issuedFrom": "Sceneria początkowa",
|
"search-issuedFrom": "Sceneria początkowa",
|
||||||
"search-via": "Przez scenerię",
|
"search-via": "Przez scenerię",
|
||||||
"search-terminatingAt": "Sceneria końcowa",
|
"search-terminatingAt": "Sceneria końcowa",
|
||||||
@@ -313,8 +315,9 @@
|
|||||||
"minTwoWayInt": "SZLAKI DWUTOROWE NIEZELEKTR. WEWNĘTRZNE (MINIMUM)"
|
"minTwoWayInt": "SZLAKI DWUTOROWE NIEZELEKTR. WEWNĘTRZNE (MINIMUM)"
|
||||||
},
|
},
|
||||||
"sceneries-placeholder": "Wyszukaj scenerię",
|
"sceneries-placeholder": "Wyszukaj scenerię",
|
||||||
"authors-placeholder": "Autor scenerii (uwzględnia inne filtry)",
|
"line-numbers-placeholder": "Numery linii (oddzielone przecinkami)",
|
||||||
"projects-placeholder": "Projekt scenerii (uwzględnia inne filtry)",
|
"authors-placeholder": "Autor scenerii",
|
||||||
|
"projects-placeholder": "Projekt scenerii",
|
||||||
"search-button-title": "SZUKAJ",
|
"search-button-title": "SZUKAJ",
|
||||||
"minimum-hours-title": "POKAŻ TYLKO SCENERIE DOSTĘPNE MINIMUM DO:",
|
"minimum-hours-title": "POKAŻ TYLKO SCENERIE DOSTĘPNE MINIMUM DO:",
|
||||||
"now": "TERAZ",
|
"now": "TERAZ",
|
||||||
|
|||||||
@@ -69,7 +69,8 @@ export const initFilters = {
|
|||||||
minTwoWayInt: 0,
|
minTwoWayInt: 0,
|
||||||
minTwoWayCatenaryInt: 0,
|
minTwoWayCatenaryInt: 0,
|
||||||
authors: '',
|
authors: '',
|
||||||
projects: ''
|
projects: '',
|
||||||
|
lines: ''
|
||||||
};
|
};
|
||||||
|
|
||||||
export const sliderStates = [
|
export const sliderStates = [
|
||||||
|
|||||||
@@ -122,19 +122,27 @@ export default defineComponent({
|
|||||||
|
|
||||||
// Check the whole consist speed limit
|
// Check the whole consist speed limit
|
||||||
const vehicleMaxSpeed = stockList.reduce((acc, stockName, i) => {
|
const vehicleMaxSpeed = stockList.reduce((acc, stockName, i) => {
|
||||||
|
if (!this.apiStore.vehiclesData) return acc;
|
||||||
|
|
||||||
const [vehicleName, vehicleCargo] = stockName.split(':');
|
const [vehicleName, vehicleCargo] = stockName.split(':');
|
||||||
|
|
||||||
const vehicleData = this.apiStore.vehiclesData?.find((v) => v.name == vehicleName);
|
const vehicle = this.apiStore.vehiclesData.vehicles.find((v) => v.name == vehicleName);
|
||||||
|
|
||||||
if (!vehicleData) return acc;
|
if (!vehicle) return acc;
|
||||||
|
|
||||||
let vehicleSpeed = vehicleData.group.speed;
|
const vehicleGroup = this.apiStore.vehiclesData.vehicleGroups.find(
|
||||||
|
(g) => g.id == vehicle.vehicleGroupsId
|
||||||
|
);
|
||||||
|
|
||||||
if (vehicleData.type == 'wagon-freight') {
|
if (!vehicleGroup) return acc;
|
||||||
|
|
||||||
|
let vehicleSpeed = vehicleGroup.speed;
|
||||||
|
|
||||||
|
if (vehicle.type == 'wagon-freight') {
|
||||||
isPassenger = false;
|
isPassenger = false;
|
||||||
|
|
||||||
if (vehicleCargo !== undefined && vehicleData.group.speedLoaded) {
|
if (vehicleCargo !== undefined && vehicleGroup.speedLoaded) {
|
||||||
vehicleSpeed = vehicleData.group.speedLoaded;
|
vehicleSpeed = vehicleGroup.speedLoaded;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -143,14 +151,23 @@ export default defineComponent({
|
|||||||
|
|
||||||
// Check the head vehicle speed limit
|
// Check the head vehicle speed limit
|
||||||
const headLocoName = stockList[0];
|
const headLocoName = stockList[0];
|
||||||
const headLocoVehicleData = this.apiStore.vehiclesData?.find((v) => v.name == headLocoName);
|
|
||||||
|
const headLocoVehicle = this.apiStore.vehiclesData!.vehicles.find(
|
||||||
|
(v) => v.name == headLocoName
|
||||||
|
);
|
||||||
|
|
||||||
|
const headLocoVehicleGroup = this.apiStore.vehiclesData!.vehicleGroups.find(
|
||||||
|
(g) => g.id == headLocoVehicle?.vehicleGroupsId
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!headLocoVehicleGroup) return vehicleMaxSpeed;
|
||||||
|
|
||||||
// Omit speed check for head vehicle if there's no data for it
|
// Omit speed check for head vehicle if there's no data for it
|
||||||
if (!headLocoName || !headLocoVehicleData || !headLocoVehicleData.group.massSpeeds)
|
if (!headLocoName || !headLocoVehicle || !headLocoVehicleGroup.massSpeeds)
|
||||||
return vehicleMaxSpeed;
|
return vehicleMaxSpeed;
|
||||||
|
|
||||||
const massSpeeds =
|
const massSpeeds =
|
||||||
headLocoVehicleData.group.massSpeeds[
|
headLocoVehicleGroup.massSpeeds[
|
||||||
stockList.length == 1 ? 'none' : isPassenger ? 'passenger' : 'cargo'
|
stockList.length == 1 ? 'none' : isPassenger ? 'passenger' : 'cargo'
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ export const useApiStore = defineStore('apiStore', {
|
|||||||
},
|
},
|
||||||
|
|
||||||
activeData: undefined as API.ActiveData.Response | undefined,
|
activeData: undefined as API.ActiveData.Response | undefined,
|
||||||
vehiclesData: undefined as API.Vehicles.Response | undefined,
|
vehiclesData: undefined as API.VehiclesData.Response | undefined,
|
||||||
|
|
||||||
donatorsData: [] as API.Donators.Response,
|
donatorsData: [] as API.Donators.Response,
|
||||||
sceneryData: [] as StationJSONData[],
|
sceneryData: [] as StationJSONData[],
|
||||||
@@ -111,7 +111,7 @@ export const useApiStore = defineStore('apiStore', {
|
|||||||
|
|
||||||
async fetchVehiclesInfo() {
|
async fetchVehiclesInfo() {
|
||||||
try {
|
try {
|
||||||
const response = await this.client!.get<API.Vehicles.Response>('api/getVehicles');
|
const response = await this.client!.get<API.VehiclesData.Response>('api/getVehiclesData');
|
||||||
|
|
||||||
this.vehiclesData = response.data;
|
this.vehiclesData = response.data;
|
||||||
this.dataStatuses.vehicles = response.data ? Status.Data.Loaded : Status.Data.Warning;
|
this.dataStatuses.vehicles = response.data ? Status.Data.Loaded : Status.Data.Warning;
|
||||||
|
|||||||
@@ -38,7 +38,8 @@ export const useMainStore = defineStore('mainStore', {
|
|||||||
modalLastClickedTarget: null,
|
modalLastClickedTarget: null,
|
||||||
currentLocale: 'pl',
|
currentLocale: 'pl',
|
||||||
|
|
||||||
isMigrateInfoCardOpen: false
|
isMigrateInfoCardOpen: false,
|
||||||
|
pinnedStationNames: []
|
||||||
}) as MainStoreState,
|
}) as MainStoreState,
|
||||||
|
|
||||||
actions: {
|
actions: {
|
||||||
@@ -431,7 +432,6 @@ export const useMainStore = defineStore('mainStore', {
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
name: scenery.name,
|
name: scenery.name,
|
||||||
|
|
||||||
generalInfo: {
|
generalInfo: {
|
||||||
...scenery,
|
...scenery,
|
||||||
authors: scenery.authors?.split(',').map((a) => a.trim()),
|
authors: scenery.authors?.split(',').map((a) => a.trim()),
|
||||||
@@ -446,7 +446,7 @@ export const useMainStore = defineStore('mainStore', {
|
|||||||
},
|
},
|
||||||
|
|
||||||
allStationInfo(): Station[] {
|
allStationInfo(): Station[] {
|
||||||
const onlineUnsavedStations = this.activeSceneryList
|
const onlineUnsavedStations: Station[] = this.activeSceneryList
|
||||||
.filter(
|
.filter(
|
||||||
(scenery) =>
|
(scenery) =>
|
||||||
this.stationList.findIndex((st) => st.name == scenery.name) == -1 &&
|
this.stationList.findIndex((st) => st.name == scenery.name) == -1 &&
|
||||||
|
|||||||
+21
-1
@@ -89,7 +89,8 @@ select {
|
|||||||
font-size: 1em;
|
font-size: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
input {
|
input,
|
||||||
|
select {
|
||||||
background: none;
|
background: none;
|
||||||
color: white;
|
color: white;
|
||||||
font-size: 1em;
|
font-size: 1em;
|
||||||
@@ -358,3 +359,22 @@ a.a-button {
|
|||||||
background-color: #aaa;
|
background-color: #aaa;
|
||||||
margin: 0.5em 0;
|
margin: 0.5em 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.g-checkbox {
|
||||||
|
position: relative;
|
||||||
|
display: inline-block;
|
||||||
|
|
||||||
|
user-select: none;
|
||||||
|
-webkit-user-select: none;
|
||||||
|
-moz-user-select: none;
|
||||||
|
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
input {
|
||||||
|
position: absolute;
|
||||||
|
width: 1px;
|
||||||
|
height: 1px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -61,8 +61,4 @@
|
|||||||
top: 50%;
|
top: 50%;
|
||||||
transform: translateY(-50%);
|
transform: translateY(-50%);
|
||||||
padding-right: 0.5em;
|
padding-right: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
select.search-input {
|
|
||||||
|
|
||||||
}
|
|
||||||
+46
-3
@@ -1,4 +1,4 @@
|
|||||||
import { Status, VehicleData } from './common';
|
import { Status, Vehicle, VehicleGroup } from './common';
|
||||||
|
|
||||||
export enum APIDataStatus {
|
export enum APIDataStatus {
|
||||||
OK = 'OK',
|
OK = 'OK',
|
||||||
@@ -329,8 +329,51 @@ export namespace API {
|
|||||||
export type Response = string[];
|
export type Response = string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export namespace Vehicles {
|
export namespace VehiclesData {
|
||||||
export type Response = VehicleData[];
|
export interface VehicleObject {
|
||||||
|
id: number;
|
||||||
|
name: string;
|
||||||
|
type: string;
|
||||||
|
cabinName: string | null;
|
||||||
|
restrictions: Record<string, any> | null;
|
||||||
|
vehicleGroupsId: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface VehicleGroupObject {
|
||||||
|
id: number;
|
||||||
|
name: string;
|
||||||
|
speed: number;
|
||||||
|
speedLoaded?: number;
|
||||||
|
speedLoco?: number;
|
||||||
|
length: number;
|
||||||
|
weight: number;
|
||||||
|
cargoTypes: VehicleCargo[] | null;
|
||||||
|
|
||||||
|
locoProps: {
|
||||||
|
coldStart: boolean;
|
||||||
|
doubleManned: boolean;
|
||||||
|
} | null;
|
||||||
|
|
||||||
|
massSpeeds: VehicleGroupMassSpeeds | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface VehicleGroupMassSpeeds {
|
||||||
|
passenger: Record<string, number> | null;
|
||||||
|
cargo: Record<string, number> | null;
|
||||||
|
none: number | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface VehicleCargo {
|
||||||
|
id: string;
|
||||||
|
weight: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Data {
|
||||||
|
vehicles: VehicleObject[];
|
||||||
|
vehicleGroups: VehicleGroupObject[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export type Response = Data;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
+5
-43
@@ -1,5 +1,6 @@
|
|||||||
import { RouteLocationRaw } from 'vue-router';
|
import { RouteLocationRaw } from 'vue-router';
|
||||||
import { StationJSONData } from '../store/typings';
|
import { StationJSONData } from '../store/typings';
|
||||||
|
import { API } from './api';
|
||||||
|
|
||||||
export type Availability = 'default' | 'unavailable' | 'nonPublic' | 'abandoned' | 'nonDefault';
|
export type Availability = 'default' | 'unavailable' | 'nonPublic' | 'abandoned' | 'nonDefault';
|
||||||
export type ScenerySpawnType = 'passenger' | 'freight' | 'loco' | 'all';
|
export type ScenerySpawnType = 'passenger' | 'freight' | 'loco' | 'all';
|
||||||
@@ -95,9 +96,7 @@ export interface TrainTimetableData {
|
|||||||
|
|
||||||
export interface Station {
|
export interface Station {
|
||||||
name: string;
|
name: string;
|
||||||
|
|
||||||
generalInfo?: StationGeneralInfo;
|
generalInfo?: StationGeneralInfo;
|
||||||
|
|
||||||
onlineInfo?: ActiveScenery;
|
onlineInfo?: ActiveScenery;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -107,7 +106,7 @@ export interface StationGeneralInfo {
|
|||||||
abbr: string;
|
abbr: string;
|
||||||
hash?: string;
|
hash?: string;
|
||||||
reqLevel: number;
|
reqLevel: number;
|
||||||
lines: string;
|
lines?: string;
|
||||||
project: string;
|
project: string;
|
||||||
projectUrl?: string;
|
projectUrl?: string;
|
||||||
signalType: string;
|
signalType: string;
|
||||||
@@ -171,7 +170,7 @@ export interface ActiveScenery {
|
|||||||
confirmed: number;
|
confirmed: number;
|
||||||
unconfirmed: number;
|
unconfirmed: number;
|
||||||
};
|
};
|
||||||
missingCheckpoints: string[];
|
missingCheckpoints: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ScenerySpawn {
|
export interface ScenerySpawn {
|
||||||
@@ -216,45 +215,8 @@ export interface CheckpointTrain {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Vehicles Data
|
// Vehicles Data
|
||||||
|
export type Vehicle = API.VehiclesData.VehicleObject;
|
||||||
export interface VehicleData {
|
export type VehicleGroup = API.VehiclesData.VehicleGroupObject;
|
||||||
id: number;
|
|
||||||
name: string;
|
|
||||||
type: string;
|
|
||||||
cabinName: string | null;
|
|
||||||
restrictions: Record<string, any> | null;
|
|
||||||
vehicleGroupsId: number;
|
|
||||||
group: VehiclesGroup;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface VehiclesGroup {
|
|
||||||
id: number;
|
|
||||||
name: string;
|
|
||||||
speed: number;
|
|
||||||
speedLoaded?: number;
|
|
||||||
speedLoco?: number;
|
|
||||||
length: number;
|
|
||||||
weight: number;
|
|
||||||
cargoTypes: VehicleCargo[] | null;
|
|
||||||
|
|
||||||
locoProps: {
|
|
||||||
coldStart: boolean;
|
|
||||||
doubleManned: boolean;
|
|
||||||
} | null;
|
|
||||||
|
|
||||||
massSpeeds: VehicleGroupMassSpeeds | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface VehicleGroupMassSpeeds {
|
|
||||||
passenger: Record<string, number> | null;
|
|
||||||
cargo: Record<string, number> | null;
|
|
||||||
none: number | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface VehicleCargo {
|
|
||||||
id: string;
|
|
||||||
weight: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface TooltipUserTrain {
|
export interface TooltipUserTrain {
|
||||||
driverName: string;
|
driverName: string;
|
||||||
|
|||||||
@@ -272,7 +272,7 @@ export default defineComponent({
|
|||||||
this.scrollDataLoaded = true;
|
this.scrollDataLoaded = true;
|
||||||
},
|
},
|
||||||
|
|
||||||
async fetchHistoryData() {
|
async fetchHistoryData() {
|
||||||
const queryParams: DispatchersQueryParams = {};
|
const queryParams: DispatchersQueryParams = {};
|
||||||
|
|
||||||
const dispatcherName = this.searchersValues['search-dispatcher'].trim() || undefined;
|
const dispatcherName = this.searchersValues['search-dispatcher'].trim() || undefined;
|
||||||
@@ -280,9 +280,25 @@ export default defineComponent({
|
|||||||
const dateFromString = this.searchersValues['search-date-from'].trim() || undefined;
|
const dateFromString = this.searchersValues['search-date-from'].trim() || undefined;
|
||||||
const dateToString = this.searchersValues['search-date-to'].trim() || undefined;
|
const dateToString = this.searchersValues['search-date-to'].trim() || undefined;
|
||||||
|
|
||||||
|
let dateFromISO: string | undefined = undefined;
|
||||||
|
let dateToISO: string | undefined = undefined;
|
||||||
|
|
||||||
|
if (dateFromString) {
|
||||||
|
let dateFrom = new Date(dateFromString);
|
||||||
|
dateFrom.setMinutes(dateFrom.getMinutes() + dateFrom.getTimezoneOffset());
|
||||||
|
dateFromISO = dateFrom.toISOString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dateToString) {
|
||||||
|
let dateTo = new Date(dateToString);
|
||||||
|
dateTo.setMinutes(dateTo.getMinutes() + dateTo.getTimezoneOffset());
|
||||||
|
dateToISO = dateTo.toISOString();
|
||||||
|
}
|
||||||
|
|
||||||
queryParams['dispatcherName'] = dispatcherName;
|
queryParams['dispatcherName'] = dispatcherName;
|
||||||
queryParams['dateFrom'] = dateFromString;
|
|
||||||
queryParams['dateTo'] = dateToString ? `${dateToString}T23:00:00` : undefined;
|
queryParams['dateFrom'] = dateFromISO;
|
||||||
|
queryParams['dateTo'] = dateToISO;
|
||||||
|
|
||||||
queryParams['countLimit'] = 30;
|
queryParams['countLimit'] = 30;
|
||||||
|
|
||||||
|
|||||||
@@ -132,6 +132,7 @@ interface TimetablesQueryParams {
|
|||||||
issuedFrom?: string;
|
issuedFrom?: string;
|
||||||
terminatingAt?: string;
|
terminatingAt?: string;
|
||||||
via?: string;
|
via?: string;
|
||||||
|
includesScenery?: string;
|
||||||
|
|
||||||
countFrom?: number;
|
countFrom?: number;
|
||||||
countLimit?: number;
|
countLimit?: number;
|
||||||
@@ -213,6 +214,7 @@ export default defineComponent({
|
|||||||
'search-train': '',
|
'search-train': '',
|
||||||
'search-driver': '',
|
'search-driver': '',
|
||||||
'search-dispatcher': '',
|
'search-dispatcher': '',
|
||||||
|
'search-includesScenery': '',
|
||||||
'search-issuedFrom': '',
|
'search-issuedFrom': '',
|
||||||
'search-via': '',
|
'search-via': '',
|
||||||
'search-terminatingAt': '',
|
'search-terminatingAt': '',
|
||||||
@@ -355,19 +357,25 @@ export default defineComponent({
|
|||||||
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;
|
||||||
const dateFrom = this.searchersValues['search-date-from'].trim() || undefined;
|
const dateFromString = this.searchersValues['search-date-from'].trim() || undefined;
|
||||||
|
const includesScenery = this.searchersValues['search-includesScenery'].trim() || undefined;
|
||||||
const issuedFrom = this.searchersValues['search-issuedFrom'].trim() || undefined;
|
const issuedFrom = this.searchersValues['search-issuedFrom'].trim() || undefined;
|
||||||
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;
|
||||||
|
|
||||||
let dateTo: string | undefined = undefined;
|
let dateFromISO: string | undefined = undefined;
|
||||||
|
let dateToISO: string | undefined = undefined;
|
||||||
|
|
||||||
if (dateFrom) {
|
if (dateFromString) {
|
||||||
const d = new Date(dateFrom);
|
let dateFrom = new Date(dateFromString);
|
||||||
d.setDate(d.getDate() + 1);
|
dateFrom.setMinutes(dateFrom.getMinutes() + dateFrom.getTimezoneOffset());
|
||||||
|
|
||||||
dateTo = d.toISOString().split('T')[0];
|
let dateTo = new Date(dateFrom);
|
||||||
|
dateTo.setDate(dateTo.getDate() + 1);
|
||||||
|
|
||||||
|
dateFromISO = dateFrom.toISOString();
|
||||||
|
dateToISO = dateTo.toISOString();
|
||||||
}
|
}
|
||||||
|
|
||||||
const queryParams: TimetablesQueryParams = {};
|
const queryParams: TimetablesQueryParams = {};
|
||||||
@@ -430,8 +438,9 @@ export default defineComponent({
|
|||||||
queryParams['countLimit'] = undefined;
|
queryParams['countLimit'] = undefined;
|
||||||
|
|
||||||
queryParams['authorName'] = authorName;
|
queryParams['authorName'] = authorName;
|
||||||
queryParams['dateFrom'] = dateFrom;
|
queryParams['dateFrom'] = dateFromISO;
|
||||||
queryParams['dateTo'] = dateTo;
|
queryParams['dateTo'] = dateToISO;
|
||||||
|
queryParams['includesScenery'] = includesScenery;
|
||||||
queryParams['issuedFrom'] = issuedFrom;
|
queryParams['issuedFrom'] = issuedFrom;
|
||||||
queryParams['terminatingAt'] = terminatingAt;
|
queryParams['terminatingAt'] = terminatingAt;
|
||||||
queryParams['via'] = via;
|
queryParams['via'] = via;
|
||||||
|
|||||||
+2
-2
@@ -4,7 +4,7 @@ import { VitePWA } from 'vite-plugin-pwa';
|
|||||||
import path from 'path';
|
import path from 'path';
|
||||||
|
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
server: { port: 5123, open: true },
|
server: { port: 5123, open: false },
|
||||||
preview: { port: 4001, open: false },
|
preview: { port: 4001, open: false },
|
||||||
publicDir: 'public',
|
publicDir: 'public',
|
||||||
css: {
|
css: {
|
||||||
@@ -28,7 +28,7 @@ export default defineConfig({
|
|||||||
runtimeCaching: [
|
runtimeCaching: [
|
||||||
{
|
{
|
||||||
urlPattern:
|
urlPattern:
|
||||||
/^https:\/\/stacjownik.spythere.eu\/api\/(getVehicles|getDonators|getSceneries)/i,
|
/^https:\/\/stacjownik.spythere.eu\/api\/(getVehiclesData|getDonators|getSceneries)/i,
|
||||||
handler: 'NetworkFirst',
|
handler: 'NetworkFirst',
|
||||||
options: {
|
options: {
|
||||||
cacheName: 'stacjownik-api-cache',
|
cacheName: 'stacjownik-api-cache',
|
||||||
|
|||||||
Reference in New Issue
Block a user