Compare commits
86 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 50102c6127 | |||
| 210c49fb04 | |||
| bc3db163b1 | |||
| b364586ebc | |||
| f3509ef217 | |||
| d584d99f4c | |||
| 078c941910 | |||
| 76670ceb29 | |||
| 4cdd8ea06a | |||
| 454ae138f7 | |||
| 640c5e262b | |||
| 7416d7d59f | |||
| ca1255b37e | |||
| de0a8520f3 | |||
| e51919896f | |||
| f9cd9a2a33 | |||
| 897d6d0c36 | |||
| b241b60657 | |||
| 1f1bef1cc9 | |||
| 05f8a0ca68 | |||
| 541572e415 | |||
| 61c7f15fcf | |||
| 1da1645c51 | |||
| 8dc670b631 | |||
| 3424f9a952 | |||
| 4079426506 | |||
| 35f2a5ca09 | |||
| b840a6cd46 | |||
| 1aca0f7ed1 | |||
| b009bc03e8 | |||
| baa39a5a99 | |||
| 99cbde3828 | |||
| e72b73ccf0 | |||
| 75e34d9f75 | |||
| 13aa1acc54 | |||
| 966181c977 | |||
| 06eb4bc607 | |||
| c07a16d245 | |||
| 7e67e34526 | |||
| 6cfc535dec | |||
| d072692db7 | |||
| 6255efd98d | |||
| 9c59c30f12 | |||
| 31302cc053 | |||
| 26fd0c67e4 | |||
| d98ec94a66 | |||
| 28485cc28c | |||
| b14c7a9502 | |||
| 90e78e5ac5 | |||
| 27f02e2c2b | |||
| 421ae1db6b | |||
| 90824dc0e5 | |||
| 583c2887e9 | |||
| f915094775 | |||
| d5e735b59e | |||
| 43a724bf13 | |||
| 337425d21c | |||
| 9ff9341851 | |||
| f9276f6c71 | |||
| f4f9a4729f | |||
| ed0906b63e | |||
| 2ae19123a3 | |||
| 0c2be7b927 | |||
| ffce2b572b | |||
| 1b5a26e380 | |||
| c9b681eaee | |||
| 160879adec | |||
| d6ddbe7af7 | |||
| 1c7b72ea1d | |||
| 83f5b07c7e | |||
| 77cb64e25a | |||
| 6d0cc8e7cd | |||
| 7fb4b0ae26 | |||
| ed191d597b | |||
| 5d24accb15 | |||
| 54c6850121 | |||
| fad8c40b4b | |||
| a8485b3531 | |||
| e7dcd125ec | |||
| d886f44c59 | |||
| 5ff6c0504e | |||
| 2158145ae8 | |||
| 6101c9bfb2 | |||
| bd9af9a630 | |||
| 2188dbdf9b | |||
| c4576d7802 |
@@ -1,18 +1,13 @@
|
|||||||
/* eslint-env node */
|
/* eslint-env node */
|
||||||
require('@rushstack/eslint-patch/modern-module-resolution')
|
require('@rushstack/eslint-patch/modern-module-resolution');
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
root: true,
|
root: true,
|
||||||
extends: [
|
extends: ['plugin:vue/vue3-essential', 'eslint:recommended', '@vue/eslint-config-typescript', '@vue/eslint-config-prettier/skip-formatting'],
|
||||||
'plugin:vue/vue3-essential',
|
|
||||||
'eslint:recommended',
|
|
||||||
'@vue/eslint-config-typescript',
|
|
||||||
'@vue/eslint-config-prettier/skip-formatting'
|
|
||||||
],
|
|
||||||
rules: {
|
rules: {
|
||||||
'vue/multi-word-component-names': 'off'
|
'vue/multi-word-component-names': 'off',
|
||||||
},
|
},
|
||||||
parserOptions: {
|
parserOptions: {
|
||||||
ecmaVersion: 'latest'
|
ecmaVersion: 'latest',
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
|
|||||||
@@ -0,0 +1,17 @@
|
|||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
github-releases-to-discord:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: Github Releases To Discord
|
||||||
|
uses: SethCohen/github-releases-to-discord@v1.13.1
|
||||||
|
with:
|
||||||
|
webhook_url: ${{ secrets.WEBHOOK_URL }}
|
||||||
|
color: "15844367"
|
||||||
|
footer_title: "Changelog - Pojazdownik"
|
||||||
|
footer_timestamp: true
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"semi": true,
|
||||||
|
"singleQuote": true,
|
||||||
|
"trailingComma": "es5",
|
||||||
|
"printWidth": 100
|
||||||
|
}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"hosting": {
|
"hosting": {
|
||||||
"public": "dist",
|
"public": "dist",
|
||||||
"ignore": ["firebase.json", "**/.*", "**/node_modules/**"],
|
"ignore": [],
|
||||||
"rewrites": [
|
"rewrites": [
|
||||||
{
|
{
|
||||||
"source": "**",
|
"source": "**",
|
||||||
@@ -9,4 +9,4 @@
|
|||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
{
|
{
|
||||||
"name": "pojazdownik",
|
"name": "pojazdownik",
|
||||||
"version": "1.7.2",
|
"version": "1.8.7",
|
||||||
"private": true,
|
"private": true,
|
||||||
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "vue-tsc --noEmit && vite build",
|
"build": "vue-tsc --noEmit && vite build",
|
||||||
@@ -15,21 +16,21 @@
|
|||||||
"pinia": "^2.0.17",
|
"pinia": "^2.0.17",
|
||||||
"prettier": "^3.0.3",
|
"prettier": "^3.0.3",
|
||||||
"vue": "^3.2.37",
|
"vue": "^3.2.37",
|
||||||
"vue-i18n": "9"
|
"vue-i18n": "9.11.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@rushstack/eslint-patch": "^1.3.3",
|
"@rushstack/eslint-patch": "^1.3.3",
|
||||||
"@vite-pwa/assets-generator": "^0.0.10",
|
"@vite-pwa/assets-generator": "^0.2.3",
|
||||||
"@vitejs/plugin-vue": "^4.1.0",
|
"@vitejs/plugin-vue": "^5.0.3",
|
||||||
"@vue/eslint-config-prettier": "^8.0.0",
|
"@vue/eslint-config-prettier": "^9.0.0",
|
||||||
"@vue/eslint-config-typescript": "^12.0.0",
|
"@vue/eslint-config-typescript": "^12.0.0",
|
||||||
"@vue/tsconfig": "^0.4.0",
|
"@vue/tsconfig": "^0.5.1",
|
||||||
"eslint": "^8.49.0",
|
"eslint": "^8.49.0",
|
||||||
"eslint-plugin-vue": "^9.17.0",
|
"eslint-plugin-vue": "^9.17.0",
|
||||||
"sass": "^1.59.3",
|
"sass": "^1.59.3",
|
||||||
"typescript": "^5.0.2",
|
"typescript": "^5.0.2",
|
||||||
"vite": "^4.2.1",
|
"vite": "^5.0.12",
|
||||||
"vite-plugin-pwa": "^0.16.5",
|
"vite-plugin-pwa": "^0.17.5",
|
||||||
"vue-tsc": "^1.2.0"
|
"vue-tsc": "^1.2.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="white" width="18px" height="18px"><path d="M0 0h24v24H0z" fill="none"/><path d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"/></svg>
|
||||||
|
After Width: | Height: | Size: 256 B |
|
Before Width: | Height: | Size: 1020 B After Width: | Height: | Size: 1020 B |
|
Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 1.3 KiB |
|
Before Width: | Height: | Size: 5.2 KiB After Width: | Height: | Size: 2.1 KiB |
|
Before Width: | Height: | Size: 1.6 KiB After Width: | Height: | Size: 1.6 KiB |
|
Before Width: | Height: | Size: 932 B After Width: | Height: | Size: 932 B |
|
Before Width: | Height: | Size: 953 B After Width: | Height: | Size: 953 B |
@@ -12,16 +12,17 @@ import AppContainerView from './views/AppContainerView.vue';
|
|||||||
import AppModals from './components/app/AppModals.vue';
|
import AppModals from './components/app/AppModals.vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
|
components: { ImageFullscreenPreview, AppContainerView, AppModals },
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
store: useStore(),
|
store: useStore(),
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
async created() {
|
async created() {
|
||||||
this.store.fetchStockInfoData();
|
|
||||||
this.store.handleRouting();
|
this.store.handleRouting();
|
||||||
|
this.store.setupAPIData();
|
||||||
},
|
},
|
||||||
components: { ImageFullscreenPreview, AppContainerView, AppModals },
|
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -34,7 +35,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
color: $textColor;
|
color: $textColor;
|
||||||
font-size: 1em;
|
font-size: 1em;
|
||||||
padding: 1em 0.5em;
|
padding: 0;
|
||||||
|
|
||||||
@media screen and (max-width: $breakpointMd) {
|
@media screen and (max-width: $breakpointMd) {
|
||||||
font-size: calc(0.7rem + 0.75vw);
|
font-size: calc(0.7rem + 0.75vw);
|
||||||
|
|||||||
@@ -7,9 +7,9 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent } from "vue";
|
import { defineComponent } from 'vue';
|
||||||
import { useStore } from "../../store";
|
import { useStore } from '../../store';
|
||||||
import RealStockCard from "../cards/RealStockCard.vue";
|
import RealStockCard from '../cards/RealStockCard.vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: { RealStockCard },
|
components: { RealStockCard },
|
||||||
|
|||||||
@@ -3,35 +3,32 @@
|
|||||||
<i18n-t keypath="footer.disclaimer" tag="div" class="text--grayed">
|
<i18n-t keypath="footer.disclaimer" tag="div" class="text--grayed">
|
||||||
<template #tos>
|
<template #tos>
|
||||||
<a style="color: #ccc" :href="$t('footer.tos-href')" target="_blank">
|
<a style="color: #ccc" :href="$t('footer.tos-href')" target="_blank">
|
||||||
{{ $t("footer.tos") }}
|
{{ $t('footer.tos') }}
|
||||||
</a>
|
</a>
|
||||||
</template>
|
</template>
|
||||||
</i18n-t>
|
</i18n-t>
|
||||||
|
|
||||||
<div class="text--grayed" v-if="store.stockData">
|
<div class="text--grayed" v-if="store.vehiclesData">
|
||||||
{{ $t("footer.version-check", { version: store.stockData.version }) }}
|
{{ $t('footer.version-check', { version: store.vehiclesData.simulatorVersion }) }}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
©
|
©
|
||||||
<a href="https://td2.info.pl/profile/?u=20777" target="_blank"
|
<a href="https://td2.info.pl/profile/?u=20777" target="_blank">Spythere</a>
|
||||||
>Spythere</a
|
{{ new Date().getUTCFullYear() }} | v{{ VERSION }}{{ !isOnProductionHost ? 'dev' : '' }}
|
||||||
>
|
|
||||||
{{ new Date().getUTCFullYear() }} | v{{ VERSION
|
|
||||||
}}{{ !isOnProductionHost ? "dev" : "" }}
|
|
||||||
</div>
|
</div>
|
||||||
</footer>
|
</footer>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent } from "vue";
|
import { defineComponent } from 'vue';
|
||||||
import packageInfo from "../../../package.json";
|
import packageInfo from '../../../package.json';
|
||||||
import { useStore } from "../../store";
|
import { useStore } from '../../store';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
isOnProductionHost: location.hostname == "pojazdownik-td2.web.app",
|
isOnProductionHost: location.hostname == 'pojazdownik-td2.web.app',
|
||||||
VERSION: packageInfo.version,
|
VERSION: packageInfo.version,
|
||||||
store: useStore(),
|
store: useStore(),
|
||||||
};
|
};
|
||||||
@@ -41,12 +38,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
footer {
|
footer {
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 0.25em;
|
|
||||||
|
|
||||||
text-align: center;
|
text-align: center;
|
||||||
padding: 1em 1em 0 1em;
|
padding: 1em 1em 0 1em;
|
||||||
margin-top: auto;
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -8,11 +8,11 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent } from "vue";
|
import { defineComponent } from 'vue';
|
||||||
import LogoSection from "../sections/LogoSection.vue";
|
import LogoSection from '../sections/LogoSection.vue';
|
||||||
import InputsSection from "../sections/InputsSection.vue";
|
import InputsSection from '../sections/InputsSection.vue';
|
||||||
import TrainImageSection from "../sections/TrainImageSection.vue";
|
import TrainImageSection from '../sections/TrainImageSection.vue';
|
||||||
import StockSection from "../sections/StockSection.vue";
|
import StockSection from '../sections/StockSection.vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: { LogoSection, InputsSection, TrainImageSection, StockSection },
|
components: { LogoSection, InputsSection, TrainImageSection, StockSection },
|
||||||
@@ -20,26 +20,31 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import "../../styles/global.scss";
|
@import '../../styles/global.scss';
|
||||||
|
|
||||||
main {
|
main {
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 1em;
|
gap: 1em;
|
||||||
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
max-width: 1300px;
|
max-width: 1350px;
|
||||||
min-height: 75vh;
|
|
||||||
|
|
||||||
grid-template-columns: 1fr 2fr;
|
grid-template-columns: 1fr 2fr;
|
||||||
grid-template-rows: auto 360px minmax(400px, 1fr);
|
grid-template-rows: auto 360px minmax(300px, 1fr);
|
||||||
|
|
||||||
|
background-color: darken($color: $bgColor, $amount: 5);
|
||||||
|
border-radius: 1em;
|
||||||
|
|
||||||
|
min-height: 950px;
|
||||||
|
|
||||||
|
padding: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (max-width: $breakpointMd) {
|
@media screen and (max-width: $breakpointMd) {
|
||||||
main {
|
main {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
grid-template-columns: 1fr;
|
gap: 1em;
|
||||||
grid-template-rows: 1fr;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,24 +1,16 @@
|
|||||||
<template>
|
<template>
|
||||||
<div
|
<div class="real-stock-card g-card" @keydown.esc="store.isRealStockListCardOpen = false">
|
||||||
class="real-stock-card g-card"
|
|
||||||
@keydown.esc="store.isRealStockListCardOpen = false"
|
|
||||||
>
|
|
||||||
<div class="g-card_bg" @click="store.isRealStockListCardOpen = false"></div>
|
<div class="g-card_bg" @click="store.isRealStockListCardOpen = false"></div>
|
||||||
|
|
||||||
<div class="card_content">
|
<div class="card_content">
|
||||||
<div class="card_nav">
|
<div class="card_nav">
|
||||||
<div class="top-pane">
|
<div class="top-pane">
|
||||||
<h1>
|
<h1>
|
||||||
{{ $t("realstock.title") }}
|
{{ $t('realstock.title') }}
|
||||||
<a href="https://td2.info.pl/profile/?u=17708" target="_blank"
|
<a href="https://td2.info.pl/profile/?u=17708" target="_blank">Railtrains997</a>
|
||||||
>Railtrains997</a
|
|
||||||
>
|
|
||||||
</h1>
|
</h1>
|
||||||
<button
|
<button class="btn action-exit" @click="store.isRealStockListCardOpen = false">
|
||||||
class="btn exit-btn"
|
<img src="/images/icon-exit.svg" alt="" />
|
||||||
@click="store.isRealStockListCardOpen = false"
|
|
||||||
>
|
|
||||||
⨯
|
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -31,7 +23,7 @@
|
|||||||
|
|
||||||
<datalist id="readyStockDataList">
|
<datalist id="readyStockDataList">
|
||||||
<option
|
<option
|
||||||
v-for="stock in store.readyStockList"
|
v-for="stock in store.realCompositionList"
|
||||||
:value="stock.stockId"
|
:value="stock.stockId"
|
||||||
:key="stock.name"
|
:key="stock.name"
|
||||||
>
|
>
|
||||||
@@ -55,32 +47,23 @@
|
|||||||
</option>
|
</option>
|
||||||
</datalist>
|
</datalist>
|
||||||
|
|
||||||
<button class="btn" @click="resetStockFilters">
|
<button class="btn action-reset" @click="resetStockFilters">
|
||||||
{{ $t("realstock.action-reset") }}
|
{{ $t('realstock.action-reset') }}
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<ul class="card_list" ref="list" @scroll="onListScroll">
|
<ul class="card_list" ref="list" @scroll="onListScroll">
|
||||||
<li
|
<li v-for="rStock in computedReadyStockList" :key="rStock.stockId">
|
||||||
v-for="rStock in computedReadyStockList"
|
<!-- :data-last-selected="store.ch === rStock.stockId" -->
|
||||||
:key="rStock.stockId"
|
|
||||||
:data-last-selected="store.chosenRealStockName === rStock.stockId"
|
|
||||||
>
|
|
||||||
<div
|
<div
|
||||||
class="stock-title"
|
class="stock-title"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
@click="chooseStock(rStock)"
|
@click="chooseStock(rStock)"
|
||||||
@keydown.enter="chooseStock(rStock)"
|
@keydown.enter="chooseStock(rStock)"
|
||||||
>
|
>
|
||||||
<img
|
<img class="stock-icon" :src="getIconURL(rStock.type)" :alt="rStock.type" />
|
||||||
class="stock-icon"
|
<b class="text--accent" style="margin-left: 5px"> {{ rStock.name }}</b>
|
||||||
:src="getIconURL(rStock.type)"
|
|
||||||
:alt="rStock.type"
|
|
||||||
/>
|
|
||||||
<b class="text--accent" style="margin-left: 5px">
|
|
||||||
{{ rStock.name }}</b
|
|
||||||
>
|
|
||||||
<div>{{ rStock.number }}</div>
|
<div>{{ rStock.number }}</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -111,24 +94,19 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent } from "vue";
|
import { defineComponent } from 'vue';
|
||||||
|
|
||||||
import { useStore } from "../../store";
|
import { useStore } from '../../store';
|
||||||
import imageMixin from "../../mixins/imageMixin";
|
import imageMixin from '../../mixins/imageMixin';
|
||||||
import stockMixin from "../../mixins/stockMixin";
|
import stockMixin from '../../mixins/stockMixin';
|
||||||
|
|
||||||
import { IReadyStockItem } from "../../types";
|
import { IRealComposition, VehicleGroupType } from '../../types';
|
||||||
import http from "../../http";
|
|
||||||
|
|
||||||
interface ResponseJSONData {
|
function getVehicleType(stockType: string): VehicleGroupType {
|
||||||
[key: string]: string;
|
if (/^E/.test(stockType)) return 'loco-electric';
|
||||||
}
|
if (/^S/.test(stockType)) return 'loco-diesel';
|
||||||
|
|
||||||
function getVehicleType(stockType: string) {
|
return 'wagon-passenger';
|
||||||
if (/^E/.test(stockType)) return "loco-e";
|
|
||||||
if (/^S/.test(stockType)) return "loco-s";
|
|
||||||
|
|
||||||
return "car-passenger";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
@@ -136,55 +114,51 @@ export default defineComponent({
|
|||||||
|
|
||||||
data: () => ({
|
data: () => ({
|
||||||
store: useStore(),
|
store: useStore(),
|
||||||
responseStatus: "loading",
|
responseStatus: 'loading',
|
||||||
isMobile:
|
isMobile:
|
||||||
"ontouchstart" in document.documentElement &&
|
'ontouchstart' in document.documentElement && navigator.userAgent.match(/Mobi/)
|
||||||
navigator.userAgent.match(/Mobi/)
|
|
||||||
? true
|
? true
|
||||||
: false,
|
: false,
|
||||||
observer: null as IntersectionObserver | null,
|
observer: null as IntersectionObserver | null,
|
||||||
searchedReadyStockName: "",
|
searchedReadyStockName: '',
|
||||||
searchedReadyStockString: "",
|
searchedReadyStockString: '',
|
||||||
visibleIndexesTo: 0,
|
visibleIndexesTo: 0,
|
||||||
lastSelectedStockId: null as string | null,
|
lastSelectedStockId: null as string | null,
|
||||||
scrollTop: 0,
|
scrollTop: 0,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
async mounted() {
|
mounted() {
|
||||||
this.mountObserver();
|
this.mountObserver();
|
||||||
this.fetchStockListData();
|
|
||||||
},
|
},
|
||||||
|
|
||||||
activated() {
|
activated() {
|
||||||
(this.$refs["focus"] as HTMLElement).focus();
|
(this.$refs['focus'] as HTMLElement).focus();
|
||||||
|
|
||||||
(this.$refs["list"] as HTMLElement).scrollTo({
|
(this.$refs['list'] as HTMLElement).scrollTo({
|
||||||
top: this.scrollTop,
|
top: this.scrollTop,
|
||||||
behavior: "auto",
|
behavior: 'auto',
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
computedReadyStockList() {
|
computedReadyStockList(): IRealComposition[] {
|
||||||
if (this.searchedReadyStockName == null) return this.store.readyStockList;
|
return this.store.realCompositionList
|
||||||
|
|
||||||
return this.store.readyStockList
|
|
||||||
.filter(
|
.filter(
|
||||||
(rs) =>
|
(rc) =>
|
||||||
rs.stockId
|
rc.stockId
|
||||||
.toLocaleLowerCase()
|
.toLocaleLowerCase()
|
||||||
.includes(this.searchedReadyStockName.toLocaleLowerCase()) &&
|
.includes(this.searchedReadyStockName.toLocaleLowerCase()) &&
|
||||||
rs.stockString
|
rc.stockString
|
||||||
.toLocaleLowerCase()
|
.toLocaleLowerCase()
|
||||||
.includes(this.searchedReadyStockString.toLocaleLowerCase()),
|
.includes(this.searchedReadyStockString.toLocaleLowerCase())
|
||||||
)
|
)
|
||||||
.filter((_, i) => i <= this.visibleIndexesTo);
|
.filter((_, i) => i <= this.visibleIndexesTo);
|
||||||
},
|
},
|
||||||
|
|
||||||
computedAvailableStockTypes() {
|
computedAvailableStockTypes() {
|
||||||
return this.store.readyStockList
|
return this.store.realCompositionList
|
||||||
.reduce((acc, rs) => {
|
.reduce((acc, rs) => {
|
||||||
rs.stockString.split(";").forEach((s) => {
|
rs.stockString.split(';').forEach((s) => {
|
||||||
if (!acc.includes(s)) acc.push(s);
|
if (!acc.includes(s)) acc.push(s);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -198,7 +172,7 @@ export default defineComponent({
|
|||||||
computedReadyStockList(curr, prev) {
|
computedReadyStockList(curr, prev) {
|
||||||
if (curr.length < prev.length) {
|
if (curr.length < prev.length) {
|
||||||
this.visibleIndexesTo = 20;
|
this.visibleIndexesTo = 20;
|
||||||
(this.$refs["list"] as HTMLElement).scrollTo({
|
(this.$refs['list'] as HTMLElement).scrollTo({
|
||||||
top: 0,
|
top: 0,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -206,41 +180,12 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
async fetchStockListData() {
|
|
||||||
const readyStockJSONData = (
|
|
||||||
await http.get<ResponseJSONData>("td2/data/readyStock.json")
|
|
||||||
).data;
|
|
||||||
|
|
||||||
if (!readyStockJSONData) {
|
|
||||||
this.responseStatus = "error";
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (let stockKey in readyStockJSONData) {
|
|
||||||
const [type, number, ...name] = stockKey.split(" ");
|
|
||||||
|
|
||||||
const obj = {
|
|
||||||
number: number.replace(/_/g, "/"),
|
|
||||||
name: name.join(" "),
|
|
||||||
stockString: readyStockJSONData[stockKey],
|
|
||||||
type,
|
|
||||||
};
|
|
||||||
|
|
||||||
this.store.readyStockList.push({
|
|
||||||
...obj,
|
|
||||||
stockId: `${obj.type} ${obj.number} ${obj.name}`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
this.responseStatus = "loaded";
|
|
||||||
},
|
|
||||||
|
|
||||||
mountObserver() {
|
mountObserver() {
|
||||||
this.observer = new IntersectionObserver((entries) => {
|
this.observer = new IntersectionObserver((entries) => {
|
||||||
if (entries[0].intersectionRatio > 0) this.visibleIndexesTo += 20;
|
if (entries[0].intersectionRatio > 0) this.visibleIndexesTo += 20;
|
||||||
});
|
});
|
||||||
|
|
||||||
this.observer.observe(this.$refs["bottom"] as HTMLElement);
|
this.observer.observe(this.$refs['bottom'] as HTMLElement);
|
||||||
},
|
},
|
||||||
|
|
||||||
getImageUrl(name: string) {
|
getImageUrl(name: string) {
|
||||||
@@ -248,11 +193,11 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
resetStockFilters() {
|
resetStockFilters() {
|
||||||
this.searchedReadyStockName = "";
|
this.searchedReadyStockName = '';
|
||||||
this.searchedReadyStockString = "";
|
this.searchedReadyStockString = '';
|
||||||
},
|
},
|
||||||
|
|
||||||
chooseStock(stockItem: IReadyStockItem) {
|
chooseStock(stockItem: IRealComposition) {
|
||||||
this.loadStockFromString(stockItem.stockString);
|
this.loadStockFromString(stockItem.stockString);
|
||||||
this.lastSelectedStockId = stockItem.stockId;
|
this.lastSelectedStockId = stockItem.stockId;
|
||||||
this.store.isRealStockListCardOpen = false;
|
this.store.isRealStockListCardOpen = false;
|
||||||
@@ -261,7 +206,7 @@ export default defineComponent({
|
|||||||
onStockItemError(e: Event, stockType: string) {
|
onStockItemError(e: Event, stockType: string) {
|
||||||
const imageEl = e.target as HTMLImageElement;
|
const imageEl = e.target as HTMLImageElement;
|
||||||
imageEl.src = `images/${getVehicleType(stockType)}-unknown.png`;
|
imageEl.src = `images/${getVehicleType(stockType)}-unknown.png`;
|
||||||
imageEl.style.opacity = "1";
|
imageEl.style.opacity = '1';
|
||||||
},
|
},
|
||||||
|
|
||||||
onListScroll(e: Event) {
|
onListScroll(e: Event) {
|
||||||
@@ -275,15 +220,17 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import "../../styles/global.scss";
|
@import '../../styles/global.scss';
|
||||||
|
|
||||||
.exit-btn {
|
.action-exit {
|
||||||
font-size: 1.2em;
|
display: flex;
|
||||||
margin: 0.25em 0;
|
background-color: #333;
|
||||||
|
border-radius: 0.25em;
|
||||||
|
padding: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn {
|
.action-reset {
|
||||||
background-color: #444;
|
background-color: #333;
|
||||||
}
|
}
|
||||||
|
|
||||||
.card_content {
|
.card_content {
|
||||||
@@ -361,7 +308,7 @@ ul {
|
|||||||
gap: 1rem;
|
gap: 1rem;
|
||||||
padding: 0.1em;
|
padding: 0.1em;
|
||||||
|
|
||||||
&[data-last-selected="true"] .stock-title {
|
&[data-last-selected='true'] .stock-title {
|
||||||
border: 1px solid $accentColor;
|
border: 1px solid $accentColor;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,72 @@
|
|||||||
|
<template>
|
||||||
|
<label>
|
||||||
|
<input type="checkbox" :data-disabled="disabled" :disabled="disabled" v-model="model" />
|
||||||
|
<div><slot /></div>
|
||||||
|
</label>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
const model = defineModel();
|
||||||
|
|
||||||
|
defineProps({
|
||||||
|
disabled: Boolean,
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
label {
|
||||||
|
text-transform: uppercase;
|
||||||
|
transition: color 200ms;
|
||||||
|
}
|
||||||
|
|
||||||
|
div {
|
||||||
|
padding: 0.25em 0.5em;
|
||||||
|
color: white;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
background-color: #222;
|
||||||
|
border-radius: 0.25em;
|
||||||
|
|
||||||
|
user-select: none;
|
||||||
|
|
||||||
|
&::before {
|
||||||
|
content: '\2716';
|
||||||
|
margin-right: 0.5em;
|
||||||
|
color: #aaa;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
input {
|
||||||
|
position: absolute;
|
||||||
|
clip: rect(1px, 1px, 1px, 1px);
|
||||||
|
padding: 0;
|
||||||
|
border: 0;
|
||||||
|
height: 1px;
|
||||||
|
width: 1px;
|
||||||
|
overflow: hidden;
|
||||||
|
|
||||||
|
&:focus-visible + div {
|
||||||
|
outline: 1px solid white;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:checked + div {
|
||||||
|
color: palegreen;
|
||||||
|
|
||||||
|
&::before {
|
||||||
|
color: palegreen;
|
||||||
|
content: '\2714';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&:disabled {
|
||||||
|
user-select: none;
|
||||||
|
-moz-user-select: none;
|
||||||
|
-webkit-user-select: none;
|
||||||
|
|
||||||
|
+ div {
|
||||||
|
opacity: 0.55;
|
||||||
|
cursor: auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -9,7 +9,7 @@
|
|||||||
v-for="locoType in locomotiveTypeList"
|
v-for="locoType in locomotiveTypeList"
|
||||||
:key="locoType.id"
|
:key="locoType.id"
|
||||||
class="btn btn--choice"
|
class="btn btn--choice"
|
||||||
:data-selected="locoType.id == store.chosenLocoPower"
|
:data-selected="locoType.id == store.chosenLocoGroup"
|
||||||
@click="selectLocoType(locoType.id)"
|
@click="selectLocoType(locoType.id)"
|
||||||
>
|
>
|
||||||
{{ $t(`inputs.${locoType.id}`) }}
|
{{ $t(`inputs.${locoType.id}`) }}
|
||||||
@@ -27,7 +27,10 @@
|
|||||||
<option :value="null" disabled>
|
<option :value="null" disabled>
|
||||||
{{ $t('inputs.input-vehicle') }}
|
{{ $t('inputs.input-vehicle') }}
|
||||||
</option>
|
</option>
|
||||||
<option v-for="loco in locoOptions" :value="loco" :key="loco.type">{{ loco.type }}<b v-if="loco.isSponsorsOnly">*</b></option>
|
<option v-for="loco in locoOptions" :value="loco" :key="loco.type">
|
||||||
|
{{ loco.type
|
||||||
|
}}<b v-if="loco.sponsorOnlyTimestamp && loco.sponsorOnlyTimestamp > Date.now()">*</b>
|
||||||
|
</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -37,7 +40,7 @@
|
|||||||
v-for="carType in carTypeList"
|
v-for="carType in carTypeList"
|
||||||
:key="carType.id"
|
:key="carType.id"
|
||||||
class="btn btn--choice"
|
class="btn btn--choice"
|
||||||
:data-selected="carType.id == store.chosenCarUseType"
|
:data-selected="carType.id == store.chosenCarGroup"
|
||||||
@click="selectCarWagonType(carType.id)"
|
@click="selectCarWagonType(carType.id)"
|
||||||
>
|
>
|
||||||
{{ $t(`inputs.${carType.id}`) }}
|
{{ $t(`inputs.${carType.id}`) }}
|
||||||
@@ -56,7 +59,10 @@
|
|||||||
{{ $t('inputs.input-carwagon') }}
|
{{ $t('inputs.input-carwagon') }}
|
||||||
</option>
|
</option>
|
||||||
|
|
||||||
<option v-for="car in carOptions" :value="car" :key="car.type">{{ car.type }}<b v-if="car.isSponsorsOnly">*</b></option>
|
<option v-for="car in carOptions" :value="car" :key="car.type">
|
||||||
|
{{ car.type
|
||||||
|
}}<b v-if="car.sponsorOnlyTimestamp && car.sponsorOnlyTimestamp > Date.now()">*</b>
|
||||||
|
</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -65,7 +71,9 @@
|
|||||||
<select
|
<select
|
||||||
id="cargo-select"
|
id="cargo-select"
|
||||||
:disabled="
|
:disabled="
|
||||||
(store.chosenCar && !store.chosenCar.loadable) || (store.chosenCar && store.chosenCar.useType == 'car-passenger') || !store.chosenCar
|
(store.chosenCar && !store.chosenCar.loadable) ||
|
||||||
|
(store.chosenCar && store.chosenCar.group == 'wagon-passenger') ||
|
||||||
|
!store.chosenCar
|
||||||
"
|
"
|
||||||
data-select="cargo"
|
data-select="cargo"
|
||||||
data-ignore-outside="1"
|
data-ignore-outside="1"
|
||||||
@@ -78,9 +86,10 @@
|
|||||||
<option :value="null" v-if="!store.chosenCar || !store.chosenCar.loadable">
|
<option :value="null" v-if="!store.chosenCar || !store.chosenCar.loadable">
|
||||||
{{ $t('inputs.no-cargo-available') }}
|
{{ $t('inputs.no-cargo-available') }}
|
||||||
</option>
|
</option>
|
||||||
|
|
||||||
<option :value="null" v-else>{{ $t('inputs.cargo-empty') }}</option>
|
<option :value="null" v-else>{{ $t('inputs.cargo-empty') }}</option>
|
||||||
|
|
||||||
<option v-for="cargo in store.chosenCar?.cargoList" :value="cargo" :key="cargo.id">
|
<option v-for="cargo in store.chosenCar?.cargoTypes" :value="cargo" :key="cargo.id">
|
||||||
{{ cargo.id }}
|
{{ cargo.id }}
|
||||||
</option>
|
</option>
|
||||||
</select>
|
</select>
|
||||||
@@ -90,7 +99,12 @@
|
|||||||
<button class="btn" @click="addVehicle(store.chosenVehicle, store.chosenCargo)">
|
<button class="btn" @click="addVehicle(store.chosenVehicle, store.chosenCargo)">
|
||||||
{{ $t('inputs.action-add') }}
|
{{ $t('inputs.action-add') }}
|
||||||
</button>
|
</button>
|
||||||
<button class="btn" @click="switchVehicles" :disabled="store.chosenStockListIndex == -1" :data-disabled="store.chosenStockListIndex == -1">
|
<button
|
||||||
|
class="btn"
|
||||||
|
@click="switchVehicles"
|
||||||
|
:disabled="store.chosenStockListIndex == -1"
|
||||||
|
:data-disabled="store.chosenStockListIndex == -1"
|
||||||
|
>
|
||||||
{{ $t('inputs.action-swap') }}
|
{{ $t('inputs.action-swap') }}
|
||||||
<b class="text--accent">
|
<b class="text--accent">
|
||||||
{{ store.chosenStockListIndex == -1 ? '' : `${store.chosenStockListIndex + 1}.` }}
|
{{ store.chosenStockListIndex == -1 ? '' : `${store.chosenStockListIndex + 1}.` }}
|
||||||
@@ -112,6 +126,7 @@ import imageMixin from '../../mixins/imageMixin';
|
|||||||
import { useStore } from '../../store';
|
import { useStore } from '../../store';
|
||||||
import stockPreviewMixin from '../../mixins/stockPreviewMixin';
|
import stockPreviewMixin from '../../mixins/stockPreviewMixin';
|
||||||
import stockMixin from '../../mixins/stockMixin';
|
import stockMixin from '../../mixins/stockMixin';
|
||||||
|
import { LocoGroupType, WagonGroupType } from '../../types';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
mixins: [imageMixin, stockPreviewMixin, stockMixin],
|
mixins: [imageMixin, stockPreviewMixin, stockMixin],
|
||||||
@@ -120,33 +135,33 @@ export default defineComponent({
|
|||||||
store: useStore(),
|
store: useStore(),
|
||||||
locomotiveTypeList: [
|
locomotiveTypeList: [
|
||||||
{
|
{
|
||||||
id: 'loco-e',
|
id: 'loco-electric',
|
||||||
desc: 'ELEKTRYCZNE',
|
desc: 'ELEKTRYCZNE',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'loco-s',
|
id: 'loco-diesel',
|
||||||
desc: 'SPALINOWE',
|
desc: 'SPALINOWE',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'loco-ezt',
|
id: 'unit-electric',
|
||||||
desc: 'ELEKTR. ZESPOŁY TRAKCYJNE',
|
desc: 'ELEKTR. ZESPOŁY TRAKCYJNE',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'loco-szt',
|
id: 'unit-diesel',
|
||||||
desc: 'SPAL. ZESPOŁY TRAKCYJNE',
|
desc: 'SPAL. ZESPOŁY TRAKCYJNE',
|
||||||
},
|
},
|
||||||
],
|
] as { id: LocoGroupType; desc: string }[],
|
||||||
|
|
||||||
carTypeList: [
|
carTypeList: [
|
||||||
{
|
{
|
||||||
id: 'car-passenger',
|
id: 'wagon-passenger',
|
||||||
desc: 'PASAŻERSKIE',
|
desc: 'PASAŻERSKIE',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'car-cargo',
|
id: 'wagon-freight',
|
||||||
desc: 'TOWAROWE',
|
desc: 'TOWAROWE',
|
||||||
},
|
},
|
||||||
],
|
] as { id: WagonGroupType; desc: string }[],
|
||||||
}),
|
}),
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
@@ -154,14 +169,14 @@ export default defineComponent({
|
|||||||
return this.store.locoDataList
|
return this.store.locoDataList
|
||||||
.slice()
|
.slice()
|
||||||
.sort((a, b) => (a.type > b.type ? 1 : -1))
|
.sort((a, b) => (a.type > b.type ? 1 : -1))
|
||||||
.filter((loco) => loco.power == this.store.chosenLocoPower);
|
.filter((loco) => loco.group == this.store.chosenLocoGroup);
|
||||||
},
|
},
|
||||||
|
|
||||||
carOptions() {
|
carOptions() {
|
||||||
return this.store.carDataList
|
return this.store.carDataList
|
||||||
.slice()
|
.slice()
|
||||||
.sort((a, b) => (a.type > b.type ? 1 : -1))
|
.sort((a, b) => (a.type > b.type ? 1 : -1))
|
||||||
.filter((car) => car.useType == this.store.chosenCarUseType);
|
.filter((car) => car.group == this.store.chosenCarGroup);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -173,17 +188,15 @@ export default defineComponent({
|
|||||||
addOrSwitchVehicle() {
|
addOrSwitchVehicle() {
|
||||||
if (!this.store.chosenVehicle) return;
|
if (!this.store.chosenVehicle) return;
|
||||||
|
|
||||||
if (this.store.chosenStockListIndex == -1) this.addVehicle(this.store.chosenVehicle, this.store.chosenCargo);
|
if (this.store.chosenStockListIndex == -1)
|
||||||
|
this.addVehicle(this.store.chosenVehicle, this.store.chosenCargo);
|
||||||
else this.switchVehicles();
|
else this.switchVehicles();
|
||||||
},
|
},
|
||||||
|
|
||||||
removeVehicle() {
|
removeVehicle() {
|
||||||
if (this.store.stockList.length == 0) return;
|
if (this.store.stockList.length == 0) return;
|
||||||
|
|
||||||
const lastStock = this.store.stockList.slice(-1)[0];
|
this.store.stockList.splice(-1);
|
||||||
|
|
||||||
if (lastStock.count > 1) lastStock.count--;
|
|
||||||
else this.store.stockList.splice(-1);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
switchVehicles() {
|
switchVehicles() {
|
||||||
@@ -197,14 +210,14 @@ export default defineComponent({
|
|||||||
this.store.stockList[this.store.chosenStockListIndex] = stockObject;
|
this.store.stockList[this.store.chosenStockListIndex] = stockObject;
|
||||||
},
|
},
|
||||||
|
|
||||||
selectLocoType(locoTypeId: string) {
|
selectLocoType(locoGroupType: LocoGroupType) {
|
||||||
this.store.chosenLocoPower = locoTypeId;
|
this.store.chosenLocoGroup = locoGroupType;
|
||||||
this.store.chosenVehicle = this.locoOptions[0];
|
this.store.chosenVehicle = this.locoOptions[0];
|
||||||
this.store.chosenLoco = this.locoOptions[0];
|
this.store.chosenLoco = this.locoOptions[0];
|
||||||
},
|
},
|
||||||
|
|
||||||
selectCarWagonType(carWagonTypeId: string) {
|
selectCarWagonType(wagonGroupType: WagonGroupType) {
|
||||||
this.store.chosenCarUseType = carWagonTypeId;
|
this.store.chosenCarGroup = wagonGroupType;
|
||||||
this.store.chosenVehicle = this.carOptions[0];
|
this.store.chosenVehicle = this.carOptions[0];
|
||||||
this.store.chosenCar = this.carOptions[0];
|
this.store.chosenCar = this.carOptions[0];
|
||||||
this.store.chosenCargo = null;
|
this.store.chosenCargo = null;
|
||||||
@@ -216,7 +229,10 @@ export default defineComponent({
|
|||||||
|
|
||||||
this.store.chosenVehicle = type == 'loco' ? this.store.chosenLoco : this.store.chosenCar;
|
this.store.chosenVehicle = type == 'loco' ? this.store.chosenLoco : this.store.chosenCar;
|
||||||
|
|
||||||
this.store.chosenCargo = this.store.chosenCar?.cargoList.find((cargo) => cargo.id == this.store.chosenCargo?.id) || null;
|
this.store.chosenCargo =
|
||||||
|
this.store.chosenCar?.cargoTypes.find(
|
||||||
|
(cargo) => cargo.id == this.store.chosenCargo?.id
|
||||||
|
) || null;
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,10 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<section class="logo-section">
|
<section class="logo-section">
|
||||||
<img
|
<img :src="`/logo-${$i18n.locale}.svg`" alt="logo pojazdownik" @click="navigate" />
|
||||||
:src="`/logo-${$i18n.locale}.svg`"
|
|
||||||
alt="logo pojazdownik"
|
|
||||||
@click="navigate"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<div class="actions">
|
<div class="actions">
|
||||||
<button
|
<button
|
||||||
@@ -26,36 +22,35 @@ export default {
|
|||||||
return {
|
return {
|
||||||
localeActions: [
|
localeActions: [
|
||||||
{
|
{
|
||||||
name: "POLSKI",
|
name: 'POLSKI',
|
||||||
locale: "pl",
|
locale: 'pl',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "ENGLISH",
|
name: 'ENGLISH',
|
||||||
locale: "en",
|
locale: 'en',
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
navigate() {
|
navigate() {
|
||||||
window.location.pathname = "";
|
window.location.pathname = '';
|
||||||
},
|
},
|
||||||
|
|
||||||
chooseLocale(locale: string) {
|
chooseLocale(locale: string) {
|
||||||
this.$i18n.locale = locale;
|
this.$i18n.locale = locale;
|
||||||
window.localStorage.setItem("locale", locale);
|
window.localStorage.setItem('locale', locale);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import "../../styles/global.scss";
|
@import '../../styles/global.scss';
|
||||||
|
|
||||||
.logo-section {
|
.logo-section {
|
||||||
grid-row: 1;
|
grid-row: 1;
|
||||||
grid-column: 1;
|
grid-column: 1;
|
||||||
margin-bottom: 1.5em;
|
|
||||||
|
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -69,7 +64,7 @@ export default {
|
|||||||
display: flex;
|
display: flex;
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
|
|
||||||
button[data-selected="true"] {
|
button[data-selected='true'] {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: $accentColor;
|
color: $accentColor;
|
||||||
text-decoration: underline;
|
text-decoration: underline;
|
||||||
|
|||||||
@@ -35,16 +35,22 @@ const sectionButtonRefs = ref([]);
|
|||||||
const store = useStore();
|
const store = useStore();
|
||||||
type SectionMode = typeof store.stockSectionMode;
|
type SectionMode = typeof store.stockSectionMode;
|
||||||
|
|
||||||
const sectionModes: SectionMode[] = ['stock-list', 'wiki-list', 'number-generator', 'stock-generator'];
|
const sectionModes: SectionMode[] = [
|
||||||
|
'stock-list',
|
||||||
|
'wiki-list',
|
||||||
|
'number-generator',
|
||||||
|
'stock-generator',
|
||||||
|
];
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
window.addEventListener('keydown', (e) => {
|
window.addEventListener('keydown', (e) => {
|
||||||
if (e.target instanceof HTMLInputElement) return;
|
if (e.target instanceof HTMLInputElement) return;
|
||||||
|
|
||||||
if (/[1234]/.test(e.key)) {
|
if (/^[1234]$/.test(e.key)) {
|
||||||
const keyNum = Number(e.key);
|
const keyNum = Number(e.key);
|
||||||
|
|
||||||
store.stockSectionMode = sectionModes[keyNum - 1];
|
store.stockSectionMode = sectionModes[keyNum - 1];
|
||||||
(sectionButtonRefs.value[keyNum - 1] as HTMLButtonElement).focus();
|
(sectionButtonRefs.value[keyNum - 1] as HTMLButtonElement)?.focus();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,53 +1,89 @@
|
|||||||
<template>
|
<template>
|
||||||
<section class="train-image-section">
|
<section class="train-image-section">
|
||||||
<div class="train-image__content" :class="{ sponsor: store.chosenVehicle?.isSponsorsOnly }">
|
<div v-if="store.chosenVehicle">
|
||||||
<img
|
<div class="image-wrapper">
|
||||||
tabindex="0"
|
<img
|
||||||
:src="store.chosenVehicle ? getThumbnailURL(store.chosenVehicle.type, 'small') : '/images/placeholder.jpg'"
|
:src="getThumbnailURL(store.chosenVehicle.type, 'small')"
|
||||||
@click="onImageClick"
|
:data-preview-active="store.chosenVehicle !== null"
|
||||||
@keydown.enter="onImageClick"
|
:data-sponsor-only="
|
||||||
@error="onImageError"
|
store.chosenVehicle.sponsorOnlyTimestamp &&
|
||||||
type="image/jpeg"
|
store.chosenVehicle.sponsorOnlyTimestamp > Date.now()
|
||||||
/>
|
"
|
||||||
</div>
|
:data-team-only="store.chosenVehicle.teamOnly"
|
||||||
|
@click="onImageClick"
|
||||||
|
@keydown.enter="onImageClick"
|
||||||
|
@load="onImageLoad"
|
||||||
|
@error="onImageError"
|
||||||
|
tabindex="0"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="train-image__info" v-if="store.chosenVehicle">
|
<div class="image-info">
|
||||||
<b class="text--accent">{{ store.chosenVehicle.type }}</b> •
|
<b class="text--accent">{{ store.chosenVehicle.type }}</b> •
|
||||||
<b style="color: #ccc">
|
<b style="color: #ccc">
|
||||||
{{ $t(`preview.${isLocomotive(store.chosenVehicle) ? store.chosenVehicle.power : store.chosenVehicle.useType}`) }}
|
|
||||||
</b>
|
|
||||||
|
|
||||||
<div style="color: #ccc">
|
|
||||||
<div>{{ store.chosenVehicle.length }}m | {{ store.chosenVehicle.mass }}t | {{ store.chosenVehicle.maxSpeed }} km/h</div>
|
|
||||||
|
|
||||||
<div v-if="isLocomotive(store.chosenVehicle)">{{ $t('preview.cabin') }} {{ store.chosenVehicle.cabinType }}</div>
|
|
||||||
|
|
||||||
<div v-else>
|
|
||||||
{{
|
{{
|
||||||
store.chosenVehicle.useType == 'car-cargo'
|
$t(
|
||||||
? $t(`usage.${store.chosenVehicle.constructionType}`)
|
`preview.${isTractionUnit(store.chosenVehicle) ? store.chosenVehicle.group : store.chosenVehicle.group}`
|
||||||
: `${$t('preview.construction')} ${store.chosenVehicle.constructionType}`
|
)
|
||||||
}}
|
}}
|
||||||
</div>
|
</b>
|
||||||
|
|
||||||
<b style="color: salmon" v-if="store.chosenVehicle.isSponsorsOnly">{{
|
<div style="color: #ccc">
|
||||||
$t('preview.sponsor-only', [
|
<div>
|
||||||
new Date(store.chosenVehicle.sponsorsOnlyTimestamp).toLocaleDateString($i18n.locale == 'pl' ? 'pl-PL' : 'en-GB'),
|
{{ store.chosenVehicle.length }}m |
|
||||||
])
|
{{ (store.chosenVehicle.weight / 1000).toFixed(1) }}t |
|
||||||
}}</b>
|
{{ store.chosenVehicle.maxSpeed }} km/h
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="isTractionUnit(store.chosenVehicle)">
|
||||||
|
{{ $t('preview.cabin') }} {{ store.chosenVehicle.cabinType }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else>
|
||||||
|
{{
|
||||||
|
store.chosenVehicle.group == 'wagon-freight'
|
||||||
|
? $t(`usage.${store.chosenVehicle.constructionType}`)
|
||||||
|
: `${$t('preview.construction')} ${store.chosenVehicle.constructionType}`
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<b
|
||||||
|
v-if="
|
||||||
|
store.chosenVehicle.sponsorOnlyTimestamp &&
|
||||||
|
store.chosenVehicle.sponsorOnlyTimestamp > Date.now()
|
||||||
|
"
|
||||||
|
class="sponsor-only"
|
||||||
|
>
|
||||||
|
{{
|
||||||
|
$t('preview.sponsor-only', [
|
||||||
|
new Date(store.chosenVehicle.sponsorOnlyTimestamp).toLocaleDateString(
|
||||||
|
$i18n.locale == 'pl' ? 'pl-PL' : 'en-GB'
|
||||||
|
),
|
||||||
|
])
|
||||||
|
}}
|
||||||
|
</b>
|
||||||
|
|
||||||
|
<b v-if="store.chosenVehicle.teamOnly" class="team-only">{{ $t('preview.team-only') }}</b>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="train-image__info" v-else>{{ $t('preview.desc') }}</div>
|
<div v-else>
|
||||||
|
<img src="/images/placeholder.jpg" alt="placeholder image" />
|
||||||
|
<div class="image-info">{{ $t('preview.desc') }}</div>
|
||||||
|
</div>
|
||||||
</section>
|
</section>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { computed, defineComponent } from 'vue';
|
import { computed, defineComponent } from 'vue';
|
||||||
import { useStore } from '../../store';
|
import { useStore } from '../../store';
|
||||||
import { isLocomotive } from '../../utils/vehicleUtils';
|
import { isTractionUnit } from '../../utils/vehicleUtils';
|
||||||
import { ILocomotive, Vehicle } from '../../types';
|
|
||||||
import imageMixin from '../../mixins/imageMixin';
|
import imageMixin from '../../mixins/imageMixin';
|
||||||
|
import { watch } from 'vue';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { Ref } from 'vue';
|
||||||
|
import { ref } from 'vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
mixins: [imageMixin],
|
mixins: [imageMixin],
|
||||||
@@ -60,35 +96,34 @@ export default defineComponent({
|
|||||||
|
|
||||||
setup() {
|
setup() {
|
||||||
const store = useStore();
|
const store = useStore();
|
||||||
|
const { chosenVehicle } = storeToRefs(store);
|
||||||
|
|
||||||
|
const imageStatus = ref('none') as Ref<'none' | 'loading' | 'loaded' | 'error'>;
|
||||||
|
|
||||||
|
watch(chosenVehicle, () => {
|
||||||
|
imageStatus.value = 'loading';
|
||||||
|
});
|
||||||
|
|
||||||
return {
|
return {
|
||||||
store,
|
store,
|
||||||
|
imageStatus,
|
||||||
chosenVehicle: computed(() => store.chosenVehicle),
|
chosenVehicle: computed(() => store.chosenVehicle),
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
watch: {
|
|
||||||
chosenVehicle(vehicle: Vehicle, prevVehicle: Vehicle) {
|
|
||||||
if (vehicle && vehicle.type != prevVehicle?.type) {
|
|
||||||
this.store.imageLoading = true;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
onImageLoad() {
|
isTractionUnit,
|
||||||
this.store.imageLoading = false;
|
|
||||||
},
|
|
||||||
|
|
||||||
onImageError(e: Event) {
|
onImageError(e: Event) {
|
||||||
const el = e.target as HTMLImageElement;
|
const el = e.target as HTMLImageElement;
|
||||||
if (el.src == '/images/placeholder.jpg') return;
|
if (el.src == '/images/placeholder.jpg') return;
|
||||||
|
|
||||||
el.src = '/images/placeholder.jpg';
|
el.src = '/images/placeholder.jpg';
|
||||||
|
this.imageStatus = 'error';
|
||||||
},
|
},
|
||||||
|
|
||||||
isLocomotive(vehicle: Vehicle): vehicle is ILocomotive {
|
onImageLoad(e: Event) {
|
||||||
return isLocomotive(vehicle);
|
this.imageStatus = 'loaded';
|
||||||
},
|
},
|
||||||
|
|
||||||
onImageClick(e: Event) {
|
onImageClick(e: Event) {
|
||||||
@@ -106,41 +141,65 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/global.scss';
|
@import '../../styles/global';
|
||||||
|
|
||||||
.train-image-section {
|
.train-image-section {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
|
||||||
grid-row: 3;
|
min-height: 250px;
|
||||||
grid-column: 1;
|
|
||||||
|
|
||||||
margin-top: 1em;
|
& > div {
|
||||||
height: 22em;
|
position: relative;
|
||||||
}
|
max-width: 100%;
|
||||||
|
width: 380px;
|
||||||
.train-image {
|
|
||||||
&__content {
|
|
||||||
&.sponsor img {
|
|
||||||
border: 1px solid salmon;
|
|
||||||
}
|
|
||||||
|
|
||||||
img {
|
|
||||||
max-width: 380px;
|
|
||||||
width: 100%;
|
|
||||||
height: 100%;
|
|
||||||
border: 1px solid white;
|
|
||||||
|
|
||||||
cursor: zoom-in;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.train-image__info {
|
img {
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
&[data-preview-active='true'] {
|
||||||
|
border: 1px solid white;
|
||||||
|
cursor: zoom-in;
|
||||||
|
}
|
||||||
|
|
||||||
|
&[data-sponsor-only='true'] {
|
||||||
|
border: 1px solid $sponsorColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
&[data-team-only='true'] {
|
||||||
|
border: 1px solid $teamColor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-wrapper {
|
||||||
|
position: relative;
|
||||||
|
min-height: 220px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loading-placeholder {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
|
background-color: #2b2b2b;
|
||||||
|
padding: 1em;
|
||||||
|
font-size: 1.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-info {
|
||||||
font-size: 1.1em;
|
font-size: 1.1em;
|
||||||
padding: 0.5em;
|
padding: 0.5em;
|
||||||
margin: 0.5em auto;
|
// margin: 0.5em auto;
|
||||||
line-height: 1.35;
|
line-height: 1.35;
|
||||||
|
|
||||||
width: 100%;
|
width: 100%;
|
||||||
@@ -150,17 +209,18 @@ export default defineComponent({
|
|||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Transition animations
|
.placeholder {
|
||||||
.img-message-anim {
|
height: 250px;
|
||||||
&-enter-from,
|
|
||||||
&-leave-to {
|
|
||||||
opacity: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
&-enter-active,
|
background-color: $bgColor;
|
||||||
&-leave-active {
|
}
|
||||||
transition: opacity 75ms ease-in 100ms;
|
|
||||||
}
|
.sponsor-only {
|
||||||
|
color: $sponsorColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
.team-only {
|
||||||
|
color: $teamColor;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (max-width: $breakpointMd) {
|
@media screen and (max-width: $breakpointMd) {
|
||||||
|
|||||||
@@ -1,77 +1,115 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="number-generator tab">
|
<div class="number-generator tab">
|
||||||
<div class="tab_header">
|
<div class="tab_header">
|
||||||
<h2>{{ $t("numgen.title") }}</h2>
|
<h2>{{ $t('numgen.title') }}</h2>
|
||||||
|
<h3>{{ $t('numgen.subtitle') }}</h3>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="tab_content">
|
<div class="tab_content">
|
||||||
<div class="options">
|
<div class="inputs">
|
||||||
<select v-model="chosenCategory" @change="randomizeTrainNumber()">
|
<label>
|
||||||
<option :value="null" disabled>
|
<span>{{ $t('numgen.train-category') }}</span>
|
||||||
{{ $t("numgen.train-category") }}
|
<select v-model="chosenCategory" @change="randomizeTrainNumber()">
|
||||||
</option>
|
<option :value="null" disabled>
|
||||||
<option
|
{{ $t('numgen.train-category') }}
|
||||||
v-for="(_, category) in genData.categories"
|
</option>
|
||||||
:key="category"
|
<option
|
||||||
:value="category"
|
v-for="(_, category) in genData.categoriesRules"
|
||||||
>
|
:key="category"
|
||||||
{{ $t(`numgen.categories.${category}`) }}
|
:value="category"
|
||||||
</option>
|
>
|
||||||
</select>
|
{{ $t(`numgen.categories.${category}`) }}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
|
||||||
<select v-model="beginRegionName" @change="randomizeTrainNumber()">
|
<label>
|
||||||
<option :value="null" disabled>
|
<span>{{ $t('numgen.start-region') }}</span>
|
||||||
{{ $t("numgen.start-region") }}
|
<select v-model="beginRegionName" @change="randomizeTrainNumber()">
|
||||||
</option>
|
<option :value="null" disabled>
|
||||||
<option
|
{{ $t('numgen.start-region') }}
|
||||||
v-for="(_, name) in genData.regionNumbers"
|
</option>
|
||||||
:key="name"
|
<option v-for="(_, name) in genData.regionNumbers" :key="name" :value="name">
|
||||||
:value="name"
|
{{ name }}
|
||||||
>
|
</option>
|
||||||
{{ name }}
|
</select>
|
||||||
</option>
|
</label>
|
||||||
</select>
|
|
||||||
|
|
||||||
<select v-model="endRegionName" @change="randomizeTrainNumber()">
|
<label>
|
||||||
<option :value="null" disabled>{{ $t("numgen.end-region") }}</option>
|
<span> {{ $t('numgen.end-region') }}</span>
|
||||||
<option
|
<select v-model="endRegionName" @change="randomizeTrainNumber()">
|
||||||
v-for="(_, name) in genData.regionNumbers"
|
<option :value="null" disabled>
|
||||||
:key="name"
|
{{ $t('numgen.end-region') }}
|
||||||
:value="name"
|
</option>
|
||||||
>
|
<option v-for="(_, name) in genData.regionNumbers" :key="name" :value="name">
|
||||||
{{ name }}
|
{{ name }}
|
||||||
</option>
|
</option>
|
||||||
</select>
|
</select>
|
||||||
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="generated-number" @click="copyNumber">
|
<div class="generated-number" @click="copyNumber">
|
||||||
<span v-if="trainNumber">
|
<span v-if="trainNumber">
|
||||||
{{ $t("numgen.number-info") }}
|
{{ $t('numgen.number-info') }}
|
||||||
<b class="text--accent">{{ trainNumber }}</b>
|
<b class="text--accent">{{ trainNumber }}</b>
|
||||||
</span>
|
</span>
|
||||||
<span v-else>{{ $t("numgen.warning") }}</span>
|
<span v-else>{{ $t('numgen.warning') }}</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- <div v-if="chosenCategory">
|
<div class="category-rules" v-if="chosenCategory && categoryRules && trainNumber">
|
||||||
Current numbering rules: {{ $t(`numgen.rules.${chosenCategory}`) }};
|
<!-- First & second digit (the same regions) -->
|
||||||
|
<div v-if="beginRegionName && endRegionName && beginRegionName == endRegionName">
|
||||||
|
<b>{{ $t('numgen.rules.two-first-digits') }}</b>
|
||||||
|
{{ $t('numgen.rules.from-pool') }}
|
||||||
|
<b class="text--accent">{{ genData.sameRegions[beginRegionName].join(', ') }}</b>
|
||||||
|
{{ $t('numgen.rules.for-region') }} {{ beginRegionName }}
|
||||||
|
</div>
|
||||||
|
|
||||||
<span v-if="beginRegionName && endRegionName">
|
<!-- First & second digit (different regions) -->
|
||||||
<span v-if="beginRegionName == endRegionName">
|
<div v-else>
|
||||||
pierwsze dwie cyfry:
|
<div>
|
||||||
{{ genData.sameRegions[beginRegionName].join(', ') }}
|
<b>
|
||||||
(numer w obrębie obszaru {{ beginRegionName }})
|
{{ $t('numgen.rules.first-digit') }}
|
||||||
</span>
|
<span class="text--accent">{{ trainNumber[0] }}</span>
|
||||||
|
</b>
|
||||||
|
{{ $t('numgen.rules.for-region-begin') }} {{ beginRegionName }}
|
||||||
|
</div>
|
||||||
|
|
||||||
<span v-else>
|
<div>
|
||||||
pierwsza cyfra: {{ genData.regionNumbers[beginRegionName] }}; druga cyfra:
|
<b>
|
||||||
{{ genData.regionNumbers[endRegionName] }}
|
{{ $t('numgen.rules.second-digit') }}
|
||||||
</span>
|
<span class="text--accent">{{ trainNumber[1] }} </span>
|
||||||
</span>
|
</b>
|
||||||
</div> -->
|
{{ $t('numgen.rules.for-region-end') }} {{ endRegionName }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Third digit (non-passenger only) -->
|
||||||
|
<div v-if="categoryRules[0] != null">
|
||||||
|
<b>
|
||||||
|
{{ $t('numgen.rules.third-digit') }}
|
||||||
|
<span class="text--accent">{{ categoryRules[0] }}</span>
|
||||||
|
</b>
|
||||||
|
{{ $t('numgen.rules.for-category') }} {{ chosenCategory }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Last digits -->
|
||||||
|
<div>
|
||||||
|
<b>
|
||||||
|
{{
|
||||||
|
$t(`numgen.rules.${categoryRules[1]?.length == 3 ? 'three' : 'two'}-last-digits`)
|
||||||
|
}}</b
|
||||||
|
>
|
||||||
|
{{ $t('numgen.rules.from-range') }}
|
||||||
|
<b class="text--accent">{{ categoryRules[1] }}-{{ categoryRules[2] }}</b>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr />
|
||||||
|
|
||||||
<div class="tab_links">
|
<div class="tab_links">
|
||||||
<a :href="$t('numgen.td2-wiki-link')" target="_blank">
|
<a :href="$t('numgen.td2-wiki-link')" target="_blank">
|
||||||
{{ $t("numgen.td2-wiki") }}
|
{{ $t('numgen.td2-wiki') }}
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -79,10 +117,15 @@
|
|||||||
|
|
||||||
<div class="tab_actions">
|
<div class="tab_actions">
|
||||||
<button class="btn" @click="randomizeTrainNumber(true)">
|
<button class="btn" @click="randomizeTrainNumber(true)">
|
||||||
{{ $t("numgen.action-random-region") }}
|
{{ $t('numgen.action-random-region') }}
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
|
<button class="btn" @click="randomizeCategory">
|
||||||
|
{{ $t('numgen.action-random-category') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
<button class="btn" @click="randomizeTrainNumber(false)">
|
<button class="btn" @click="randomizeTrainNumber(false)">
|
||||||
{{ $t("numgen.action-random-number") }}
|
{{ $t('numgen.action-random-number') }}
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -90,14 +133,15 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { Ref, ref } from "vue";
|
import { Ref, ref } from 'vue';
|
||||||
import { useI18n } from "vue-i18n";
|
import { useI18n } from 'vue-i18n';
|
||||||
|
|
||||||
import genData from "../../constants/numberGeneratorData.json";
|
import genData from '../../constants/numberGeneratorData.json';
|
||||||
|
import { computed } from 'vue';
|
||||||
|
|
||||||
const i18n = useI18n();
|
const i18n = useI18n();
|
||||||
type RegionName = keyof typeof genData.regionNumbers;
|
type RegionName = keyof typeof genData.regionNumbers;
|
||||||
type Category = keyof typeof genData.categories;
|
type Category = keyof typeof genData.categoriesRules;
|
||||||
|
|
||||||
const beginRegionName = ref(null) as Ref<RegionName | null>;
|
const beginRegionName = ref(null) as Ref<RegionName | null>;
|
||||||
const endRegionName = ref(null) as Ref<RegionName | null>;
|
const endRegionName = ref(null) as Ref<RegionName | null>;
|
||||||
@@ -108,27 +152,37 @@ const trainNumber = ref(null) as Ref<string | null>;
|
|||||||
const copyNumber = () => {
|
const copyNumber = () => {
|
||||||
if (trainNumber.value) {
|
if (trainNumber.value) {
|
||||||
navigator.clipboard.writeText(trainNumber.value);
|
navigator.clipboard.writeText(trainNumber.value);
|
||||||
alert(i18n.t("numgen.alert"));
|
alert(i18n.t('numgen.alert'));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const categoryRules = computed(() => {
|
||||||
|
if (!chosenCategory.value) return null;
|
||||||
|
|
||||||
|
return genData.categoriesRules[chosenCategory.value];
|
||||||
|
});
|
||||||
|
|
||||||
|
const randomizeCategory = () => {
|
||||||
|
const categoryKeys = Object.keys(genData.categoriesRules) as Category[];
|
||||||
|
chosenCategory.value = categoryKeys[~~(Math.random() * categoryKeys.length)];
|
||||||
|
|
||||||
|
randomizeTrainNumber(false);
|
||||||
|
};
|
||||||
|
|
||||||
const randomizeTrainNumber = (randomizeRegions = false) => {
|
const randomizeTrainNumber = (randomizeRegions = false) => {
|
||||||
// if (categoryRules.value == null) return;
|
// if (categoryRules.value == null) return;
|
||||||
|
|
||||||
const regionKeys = Object.keys(genData.regionNumbers);
|
const regionKeys = Object.keys(genData.regionNumbers);
|
||||||
|
|
||||||
if (beginRegionName.value == null || randomizeRegions)
|
if (beginRegionName.value == null || randomizeRegions)
|
||||||
beginRegionName.value = regionKeys[
|
beginRegionName.value = regionKeys[(regionKeys.length * Math.random()) << 0] as RegionName;
|
||||||
(regionKeys.length * Math.random()) << 0
|
|
||||||
] as RegionName;
|
|
||||||
|
|
||||||
if (endRegionName.value == null || randomizeRegions)
|
if (endRegionName.value == null || randomizeRegions)
|
||||||
endRegionName.value = regionKeys[
|
endRegionName.value = regionKeys[(regionKeys.length * Math.random()) << 0] as RegionName;
|
||||||
(regionKeys.length * Math.random()) << 0
|
|
||||||
] as RegionName;
|
|
||||||
|
|
||||||
let number = "";
|
let number = '';
|
||||||
|
|
||||||
|
// Two first numbers (begin & end regions)
|
||||||
if (beginRegionName.value == endRegionName.value) {
|
if (beginRegionName.value == endRegionName.value) {
|
||||||
const sameRegionsNumbers = genData.sameRegions[beginRegionName.value!];
|
const sameRegionsNumbers = genData.sameRegions[beginRegionName.value!];
|
||||||
const randRegionNumber =
|
const randRegionNumber =
|
||||||
@@ -147,48 +201,57 @@ const randomizeTrainNumber = (randomizeRegions = false) => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (chosenCategory.value == null) chosenCategory.value = "EI";
|
// Choose default category if it's not chosen
|
||||||
|
if (chosenCategory.value == null) chosenCategory.value = 'EI';
|
||||||
|
|
||||||
const rulesArray = genData.categories[chosenCategory.value]
|
// Get category rules
|
||||||
.split(";")
|
const [thirdNumber, minRange, maxRange] = categoryRules.value!;
|
||||||
.map((r) => ({
|
|
||||||
index: r.split(":")[0],
|
|
||||||
rule: r.split(":")[1],
|
|
||||||
nums: Number(r.split(":")[2] || "1"),
|
|
||||||
}));
|
|
||||||
|
|
||||||
rulesArray.forEach((r) => {
|
// Third number
|
||||||
const range = r.rule.split("-");
|
number += thirdNumber ?? '';
|
||||||
|
|
||||||
if (range.length == 1) number += r.rule;
|
// Remaining numbers
|
||||||
else {
|
const rangeNums = minRange!.length;
|
||||||
const [minRange, maxRange] = range;
|
const randRange = Math.floor(
|
||||||
const randRange = Math.floor(
|
Math.random() * (Number(maxRange) - Number(minRange)) + Number(minRange)
|
||||||
Math.random() * (Number(maxRange) - Number(minRange)) +
|
).toString();
|
||||||
Number(minRange),
|
const leadingZeros = new Array(Math.abs(randRange.length - rangeNums)).fill('0').join('');
|
||||||
).toString();
|
|
||||||
|
|
||||||
number +=
|
number += `${leadingZeros}${randRange}`;
|
||||||
new Array(Math.abs(randRange.length - r.nums)).fill("0").join("") +
|
|
||||||
randRange;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
trainNumber.value = number;
|
trainNumber.value = number;
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import "../../styles/tab.scss";
|
@import '../../styles/tab.scss';
|
||||||
@import "../../styles/global.scss";
|
@import '../../styles/global.scss';
|
||||||
|
|
||||||
.options {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
|
||||||
gap: 0.5em;
|
|
||||||
|
|
||||||
|
.category-select {
|
||||||
select {
|
select {
|
||||||
width: 100%;
|
width: auto;
|
||||||
|
min-width: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
margin-bottom: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.inputs {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(15em, 1fr));
|
||||||
|
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.inputs > label {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.25em;
|
||||||
|
|
||||||
|
span {
|
||||||
|
color: #ccc;
|
||||||
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -204,14 +267,19 @@ const randomizeTrainNumber = (randomizeRegions = false) => {
|
|||||||
background-color: $secondaryColor;
|
background-color: $secondaryColor;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.category-rules {
|
||||||
|
margin-bottom: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
.tab_actions {
|
.tab_actions {
|
||||||
grid-template-columns: 1fr 1fr;
|
grid-template-columns: repeat(3, 1fr);
|
||||||
margin: 0.5em 0;
|
margin: 0.5em 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tab_links {
|
.tab_links {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: flex-end;
|
justify-content: flex-end;
|
||||||
|
margin: 0.25em 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (max-width: $breakpointMd) {
|
@media screen and (max-width: $breakpointMd) {
|
||||||
@@ -221,8 +289,20 @@ const randomizeTrainNumber = (randomizeRegions = false) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (max-width: $breakpointSm) {
|
@media screen and (max-width: $breakpointSm) {
|
||||||
.options select {
|
.regions-select {
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
.regions-select select {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.category-select select {
|
||||||
|
min-width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-rules {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -12,22 +12,28 @@
|
|||||||
{{ $t('stockgen.properties-desc') }}
|
{{ $t('stockgen.properties-desc') }}
|
||||||
</b>
|
</b>
|
||||||
|
|
||||||
<div class="tab_attributes">
|
<div class="inputs">
|
||||||
<label>
|
<label>
|
||||||
{{ $t('stockgen.input-mass') }}
|
<span>{{ $t('stockgen.input-mass') }}</span>
|
||||||
<input type="number" v-model="maxMass" step="100" max="4000" min="0" />
|
<input type="number" v-model="maxTons" step="100" max="4000" min="0" />
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<label>
|
<label>
|
||||||
{{ $t('stockgen.input-length') }}
|
<span>{{ $t('stockgen.input-length') }}</span>
|
||||||
<input type="number" v-model="maxLength" step="25" max="650" min="0" />
|
<input type="number" v-model="maxLength" step="25" max="650" min="0" />
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
<label>
|
<label>
|
||||||
{{ $t('stockgen.input-carcount') }}
|
<span>{{ $t('stockgen.input-carcount') }}</span>
|
||||||
<input type="number" v-model="maxCarCount" step="1" max="60" min="1" />
|
<input type="number" v-model="maxCarCount" step="1" max="60" min="1" />
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!-- <hr style="margin: 1em 0" /> -->
|
||||||
|
|
||||||
|
<!-- <div class="generator_options">
|
||||||
|
<Checkbox v-model="isCarGroupingEnabled">Grupuj wylosowane wagony (ustawia podobne wagony obok siebie w składzie)</Checkbox>
|
||||||
|
</div> -->
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
@@ -79,15 +85,27 @@
|
|||||||
<hr />
|
<hr />
|
||||||
|
|
||||||
<div class="tab_actions">
|
<div class="tab_actions">
|
||||||
<button class="btn" :data-disabled="computedChosenCarTypes.size == 0" @click="generateStock()">
|
<button
|
||||||
|
class="btn"
|
||||||
|
:data-disabled="computedChosenCarTypes.size == 0"
|
||||||
|
@click="generateStock()"
|
||||||
|
>
|
||||||
{{ $t('stockgen.action-generate') }}
|
{{ $t('stockgen.action-generate') }}
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<button class="btn" :data-disabled="computedChosenCarTypes.size == 0" @click="generateStock(true)">
|
<button
|
||||||
|
class="btn"
|
||||||
|
:data-disabled="computedChosenCarTypes.size == 0"
|
||||||
|
@click="generateStock(true)"
|
||||||
|
>
|
||||||
{{ $t('stockgen.action-generate-empty') }}
|
{{ $t('stockgen.action-generate-empty') }}
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<button class="btn" :data-disabled="computedChosenCarTypes.size == 0" @click="resetChosenCargo">
|
<button
|
||||||
|
class="btn"
|
||||||
|
:data-disabled="computedChosenCarTypes.size == 0"
|
||||||
|
@click="resetChosenCargo"
|
||||||
|
>
|
||||||
{{ $t('stockgen.action-reset') }}
|
{{ $t('stockgen.action-reset') }}
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
@@ -101,12 +119,11 @@ import { useStore } from '../../store';
|
|||||||
|
|
||||||
import stockMixin from '../../mixins/stockMixin';
|
import stockMixin from '../../mixins/stockMixin';
|
||||||
import { ICargo, ICarWagon, IStock } from '../../types';
|
import { ICargo, ICarWagon, IStock } from '../../types';
|
||||||
import warningsMixin from '../../mixins/warningsMixin';
|
import { isTractionUnit } from '../../utils/vehicleUtils';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: 'stock-generator',
|
name: 'stock-generator',
|
||||||
|
mixins: [stockMixin],
|
||||||
mixins: [stockMixin, warningsMixin],
|
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
@@ -117,10 +134,12 @@ export default defineComponent({
|
|||||||
|
|
||||||
previewTimeout: -1,
|
previewTimeout: -1,
|
||||||
|
|
||||||
maxMass: 3000,
|
maxTons: 3000,
|
||||||
maxLength: 650,
|
maxLength: 650,
|
||||||
maxCarCount: 50,
|
maxCarCount: 50,
|
||||||
|
|
||||||
|
isCarGroupingEnabled: false,
|
||||||
|
|
||||||
store: useStore(),
|
store: useStore(),
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
@@ -131,9 +150,9 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
computedCargoData() {
|
computedCargoData() {
|
||||||
if (!this.store.stockData?.generator.cargo) return [];
|
if (!this.store.vehiclesData?.generator.cargo) return [];
|
||||||
|
|
||||||
const cargoGeneratorData = this.store.stockData.generator.cargo;
|
const cargoGeneratorData = this.store.vehiclesData.generator.cargo;
|
||||||
|
|
||||||
return Object.keys(cargoGeneratorData)
|
return Object.keys(cargoGeneratorData)
|
||||||
.sort((v1, v2) => this.$t(`cargo.${v1}`).localeCompare(this.$t(`cargo.${v2}`)))
|
.sort((v1, v2) => this.$t(`cargo.${v1}`).localeCompare(this.$t(`cargo.${v2}`)))
|
||||||
@@ -163,10 +182,21 @@ export default defineComponent({
|
|||||||
this.excludedCarTypes.length = 0;
|
this.excludedCarTypes.length = 0;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
// WIP
|
||||||
|
groupStock(stockList: IStock[]) {
|
||||||
|
if (!this.isCarGroupingEnabled) return false;
|
||||||
|
|
||||||
|
stockList.sort((s1, s2) => {
|
||||||
|
return (s1.vehicleRef.constructionType + s1.cargo?.id).localeCompare(
|
||||||
|
s2.vehicleRef.constructionType + s2.cargo?.id
|
||||||
|
);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
generateStock(empty = false) {
|
generateStock(empty = false) {
|
||||||
const generatedChosenStockList = this.chosenCargoTypes.reduce(
|
const generatedChosenStockList = this.chosenCargoTypes.reduce(
|
||||||
(acc, type) => {
|
(acc, type) => {
|
||||||
this.store.stockData?.generator.cargo[type]
|
this.store.vehiclesData?.generator.cargo[type]
|
||||||
.filter((c) => !this.excludedCarTypes.includes(c.split(':')[0]))
|
.filter((c) => !this.excludedCarTypes.includes(c.split(':')[0]))
|
||||||
.forEach((c) => {
|
.forEach((c) => {
|
||||||
const [type, cargoType] = c.split(':');
|
const [type, cargoType] = c.split(':');
|
||||||
@@ -175,12 +205,15 @@ export default defineComponent({
|
|||||||
const cargoObjs = [] as (ICargo | undefined)[];
|
const cargoObjs = [] as (ICargo | undefined)[];
|
||||||
|
|
||||||
if (!cargoType || empty) cargoObjs.push(undefined);
|
if (!cargoType || empty) cargoObjs.push(undefined);
|
||||||
else if (cargoType == 'all') cargoObjs.push(...carWagonObjs[0]!.cargoList);
|
else if (cargoType == 'all') cargoObjs.push(...carWagonObjs[0]!.cargoTypes);
|
||||||
else cargoObjs.push(carWagonObjs[0]?.cargoList.find((cargo) => cargo.id == cargoType));
|
else
|
||||||
|
cargoObjs.push(carWagonObjs[0]?.cargoTypes.find((cargo) => cargo.id == cargoType));
|
||||||
|
|
||||||
carWagonObjs.forEach((cw) => {
|
carWagonObjs.forEach((cw) => {
|
||||||
cargoObjs.forEach((cargoObj) => {
|
cargoObjs.forEach((cargoObj) => {
|
||||||
const chosenStock = acc.find((a) => a.constructionType.includes(cw.constructionType));
|
const chosenStock = acc.find((a) =>
|
||||||
|
a.constructionType.includes(cw.constructionType)
|
||||||
|
);
|
||||||
|
|
||||||
if (!chosenStock)
|
if (!chosenStock)
|
||||||
acc.push({
|
acc.push({
|
||||||
@@ -206,30 +239,38 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
|
|
||||||
for (let i = 0; i < 10; i++) {
|
for (let i = 0; i < 10; i++) {
|
||||||
const headingLoco = this.store.stockList[0]?.isLoco ? this.store.stockList[0] : undefined;
|
this.store.stockList.splice(
|
||||||
this.store.stockList.length = headingLoco ? 1 : 0;
|
this.store.stockList.length > 0 && isTractionUnit(this.store.stockList[0].vehicleRef)
|
||||||
|
? 1
|
||||||
|
: 0
|
||||||
|
);
|
||||||
|
|
||||||
const maxMass = this.store.acceptableMass > 0 ? Math.min(this.store.acceptableMass, this.maxMass) : this.maxMass;
|
let carCount = 0;
|
||||||
|
const maxWeight =
|
||||||
|
this.store.acceptableWeight > 0
|
||||||
|
? Math.min(this.store.acceptableWeight, this.maxTons * 1000)
|
||||||
|
: this.maxTons * 1000;
|
||||||
|
|
||||||
let exceeded = false;
|
// eslint-disable-next-line no-constant-condition
|
||||||
|
while (true) {
|
||||||
while (!exceeded) {
|
const randomStockType =
|
||||||
const randomStockType = generatedChosenStockList[~~(Math.random() * generatedChosenStockList.length)];
|
generatedChosenStockList[~~(Math.random() * generatedChosenStockList.length)];
|
||||||
const { carWagon, cargo } = randomStockType.carPool[~~(Math.random() * randomStockType.carPool.length)];
|
const { carWagon, cargo } =
|
||||||
|
randomStockType.carPool[~~(Math.random() * randomStockType.carPool.length)];
|
||||||
|
|
||||||
if (
|
if (
|
||||||
this.store.totalMass + (cargo?.totalMass || carWagon.mass) > maxMass ||
|
this.store.totalWeight + (carWagon.weight + (cargo?.weight ?? 0)) > maxWeight ||
|
||||||
this.store.totalLength + carWagon.length > this.maxLength ||
|
this.store.totalLength + carWagon.length > this.maxLength ||
|
||||||
this.store.stockList.length > this.maxCarCount
|
carCount >= this.maxCarCount
|
||||||
) {
|
) {
|
||||||
exceeded = true;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.addCarWagon(carWagon, cargo);
|
this.addCarWagon(carWagon, cargo);
|
||||||
|
carCount++;
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentGenerationValue = this.store.totalLength + this.store.totalMass + this.store.stockList.length;
|
const currentGenerationValue = this.store.totalLength + this.store.totalWeight + carCount;
|
||||||
|
|
||||||
if (bestGeneration.value < currentGenerationValue) {
|
if (bestGeneration.value < currentGenerationValue) {
|
||||||
bestGeneration.stockList = this.store.stockList;
|
bestGeneration.stockList = this.store.stockList;
|
||||||
@@ -237,6 +278,10 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const bestStockList = bestGeneration.stockList;
|
||||||
|
|
||||||
|
this.groupStock(bestStockList);
|
||||||
|
|
||||||
this.store.stockList = bestGeneration.stockList;
|
this.store.stockList = bestGeneration.stockList;
|
||||||
this.store.stockSectionMode = 'stock-list';
|
this.store.stockSectionMode = 'stock-list';
|
||||||
},
|
},
|
||||||
@@ -249,7 +294,7 @@ export default defineComponent({
|
|||||||
this.store.chosenLoco = null;
|
this.store.chosenLoco = null;
|
||||||
this.store.chosenCargo = null;
|
this.store.chosenCargo = null;
|
||||||
|
|
||||||
if (c) this.store.chosenCarUseType = c?.useType;
|
if (c) this.store.chosenCarGroup = c?.group;
|
||||||
},
|
},
|
||||||
|
|
||||||
toggleCargoChosen(cargoType: string, vehicles: string[]) {
|
toggleCargoChosen(cargoType: string, vehicles: string[]) {
|
||||||
@@ -283,6 +328,11 @@ export default defineComponent({
|
|||||||
@import '../../styles/global.scss';
|
@import '../../styles/global.scss';
|
||||||
@import '../../styles/tab.scss';
|
@import '../../styles/tab.scss';
|
||||||
|
|
||||||
|
h2 {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
.generator_cargo,
|
.generator_cargo,
|
||||||
.generator_vehicles {
|
.generator_vehicles {
|
||||||
display: grid;
|
display: grid;
|
||||||
@@ -323,6 +373,34 @@ export default defineComponent({
|
|||||||
gap: 1em;
|
gap: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.inputs {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(12em, 1fr));
|
||||||
|
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.inputs > label {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.25em;
|
||||||
|
|
||||||
|
span {
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type='text'] {
|
||||||
|
background-color: white;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.generator_options {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
.generator_warning {
|
.generator_warning {
|
||||||
background-color: $accentColor;
|
background-color: $accentColor;
|
||||||
padding: 0.5em;
|
padding: 0.5em;
|
||||||
|
|||||||
@@ -4,157 +4,226 @@
|
|||||||
<h2>{{ $t('stocklist.title') }}</h2>
|
<h2>{{ $t('stocklist.title') }}</h2>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="stock_actions">
|
<div class="tab_content">
|
||||||
<button class="btn btn--image" @click="clickFileInput">
|
<div class="stock_actions">
|
||||||
<input type="file" @change="uploadStock" ref="conFile" accept=".con,.txt" />
|
<button class="btn btn--image" @click="clickFileInput">
|
||||||
<img src="/images/icon-upload.svg" alt="upload icon" />
|
<input type="file" @change="uploadStock" ref="conFile" accept=".con,.txt" />
|
||||||
{{ $t('stocklist.action-upload') }}
|
<img src="/images/icon-upload.svg" alt="upload icon" />
|
||||||
</button>
|
{{ $t('stocklist.action-upload') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
<button class="btn btn--image" :data-disabled="stockIsEmpty" :disabled="stockIsEmpty" @click="downloadStock">
|
<button
|
||||||
<img src="/images/icon-download.svg" alt="download icon" />
|
class="btn btn--image"
|
||||||
{{ $t('stocklist.action-download') }}
|
:data-disabled="stockIsEmpty"
|
||||||
</button>
|
:disabled="stockIsEmpty"
|
||||||
|
@click="downloadStock"
|
||||||
<button class="btn btn--image" :data-disabled="stockIsEmpty" :disabled="stockIsEmpty" @click="copyToClipboard">
|
|
||||||
<img src="/images/icon-copy.svg" alt="copy icon" />
|
|
||||||
{{ $t('stocklist.action-copy') }}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<button class="btn btn--image" :data-disabled="stockIsEmpty" :disabled="stockIsEmpty" @click="resetStock">
|
|
||||||
<img src="/images/icon-reset.svg" alt="reset icon" />
|
|
||||||
{{ $t('stocklist.action-reset') }}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<button class="btn btn--image" :data-disabled="stockIsEmpty" :disabled="stockIsEmpty" @click="shuffleCars">
|
|
||||||
<img src="/images/icon-shuffle.svg" alt="shuffle icon" />
|
|
||||||
{{ $t('stocklist.action-shuffle') }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="stock_controls" :data-disabled="store.chosenStockListIndex == -1">
|
|
||||||
<b v-if="store.chosenStockListIndex >= 0">
|
|
||||||
{{ $t('stocklist.vehicle-no') }}
|
|
||||||
<span class="text--accent">{{ store.chosenStockListIndex + 1 }}</span>
|
|
||||||
|
|
||||||
</b>
|
|
||||||
|
|
||||||
<b v-else>
|
|
||||||
{{ $t('stocklist.no-vehicle-chosen') }}
|
|
||||||
</b>
|
|
||||||
|
|
||||||
<button class="btn" :tabindex="store.chosenStockListIndex == -1 ? -1 : 0" @click="moveUpStock(store.chosenStockListIndex)">
|
|
||||||
<img :src="getIconURL('higher')" alt="move up vehicle" />
|
|
||||||
{{ $t('stocklist.action-move-up') }}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<button class="btn" :tabindex="store.chosenStockListIndex == -1 ? -1 : 0" @click="moveDownStock(store.chosenStockListIndex)">
|
|
||||||
<img :src="getIconURL('lower')" alt="move down vehicle" />
|
|
||||||
{{ $t('stocklist.action-move-down') }}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<button class="btn" :tabindex="store.chosenStockListIndex == -1 ? -1 : 0" @click="removeStock(store.chosenStockListIndex)">
|
|
||||||
<img :src="getIconURL('remove')" alt="remove vehicle" />
|
|
||||||
{{ $t('stocklist.action-remove') }}
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="stock_specs">
|
|
||||||
<b class="real-stock-info" v-if="store.chosenRealStock">
|
|
||||||
<span class="text--accent">
|
|
||||||
<img :src="getIconURL(store.chosenRealStock.type)" :alt="store.chosenRealStock.type" />
|
|
||||||
{{ store.chosenRealStock.number }} {{ store.chosenRealStock.name }}
|
|
||||||
</span>
|
|
||||||
|
|
|
||||||
</b>
|
|
||||||
|
|
||||||
<span>
|
|
||||||
{{ $t('stocklist.mass') }}
|
|
||||||
<span class="text--accent">{{ store.totalMass }}t</span> ({{ $t('stocklist.mass-accepted') }}:
|
|
||||||
<span class="text--accent">{{ store.acceptableMass ? store.acceptableMass + 't' : '-' }}</span
|
|
||||||
>) - {{ $t('stocklist.length') }}:
|
|
||||||
<span class="text--accent">{{ store.totalLength }}m</span>
|
|
||||||
- {{ $t('stocklist.vmax') }}:
|
|
||||||
<span class="text--accent">{{ store.maxStockSpeed }} km/h</span>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="stock_spawn-settings">
|
|
||||||
<label v-if="store.stockSupportsColdStart" :data-checked="store.isColdStart">
|
|
||||||
<input type="checkbox" v-model="store.isColdStart" />
|
|
||||||
{{ $t('stocklist.coldstart-info') }}
|
|
||||||
</label>
|
|
||||||
|
|
||||||
<label v-if="store.stockSupportsDoubleManning" :data-checked="store.isDoubleManned">
|
|
||||||
<input type="checkbox" v-model="store.isDoubleManned" />
|
|
||||||
{{ $t('stocklist.doublemanning-info') }}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="stock_warnings" v-if="stockHasWarnings">
|
|
||||||
<div class="warning" v-if="locoNotSuitable">(!) {{ $t('stocklist.warning-not-suitable') }}</div>
|
|
||||||
|
|
||||||
<div class="warning" v-if="trainTooLong && store.isTrainPassenger">(!) {{ $t('stocklist.warning-passenger-too-long') }}</div>
|
|
||||||
|
|
||||||
<div class="warning" v-if="trainTooLong && !store.isTrainPassenger">(!) {{ $t('stocklist.warning-freight-too-long') }}</div>
|
|
||||||
|
|
||||||
<div class="warning" v-if="trainTooHeavy">
|
|
||||||
(!)
|
|
||||||
<i18n-t keypath="stocklist.warning-too-heavy">
|
|
||||||
<template #href>
|
|
||||||
<a target="_blank" href="https://docs.google.com/spreadsheets/d/1KVa5vn2d8XGkXQFwbavVudwKqUQxbLOucHWs2VYqAUE">
|
|
||||||
{{ $t('stocklist.acceptable-mass-docs') }}
|
|
||||||
</a>
|
|
||||||
</template>
|
|
||||||
</i18n-t>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="warning" v-if="tooManyLocomotives">
|
|
||||||
{{ $t('stocklist.warning-too-many-locos') }}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<StockThumbnails :onListItemClick="onListItemClick" />
|
|
||||||
|
|
||||||
<!-- Stock list -->
|
|
||||||
<ul ref="stock_list">
|
|
||||||
<li v-if="stockIsEmpty" class="list-empty">
|
|
||||||
<div class="stock-info">{{ $t('stocklist.list-empty') }}</div>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<TransitionGroup name="stock-list-anim">
|
|
||||||
<li
|
|
||||||
v-for="(stock, i) in store.stockList"
|
|
||||||
:key="stock.id"
|
|
||||||
:class="{ loco: stock.isLoco }"
|
|
||||||
tabindex="0"
|
|
||||||
@click="onListItemClick(i)"
|
|
||||||
@keydown.enter="onListItemClick(i)"
|
|
||||||
@keydown.w="moveUpStock(i)"
|
|
||||||
@keydown.s="moveDownStock(i)"
|
|
||||||
@keydown.backspace="removeStock(i)"
|
|
||||||
ref="itemRefs"
|
|
||||||
>
|
>
|
||||||
<div class="stock-info" @dragstart="onDragStart(i)" @drop="onDrop($event, i)" @dragover="allowDrop" draggable="true">
|
<img src="/images/icon-download.svg" alt="download icon" />
|
||||||
<span class="stock-info__no" :data-selected="i == store.chosenStockListIndex">
|
{{ $t('stocklist.action-download') }}
|
||||||
<span v-if="i == store.chosenStockListIndex">• </span>
|
</button>
|
||||||
{{ i + 1 }}.
|
|
||||||
</span>
|
|
||||||
|
|
||||||
<span class="stock-info__type" :class="{ sponsor: stock.isSponsorsOnly }">
|
<button
|
||||||
{{ stock.isLoco ? stock.type : getCarSpecFromType(stock.type) }}
|
class="btn btn--image"
|
||||||
</span>
|
:data-disabled="stockIsEmpty"
|
||||||
|
:disabled="stockIsEmpty"
|
||||||
|
@click="copyToClipboard"
|
||||||
|
>
|
||||||
|
<img src="/images/icon-copy.svg" alt="copy icon" />
|
||||||
|
{{ $t('stocklist.action-copy') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
<span class="stock-info__cargo" v-if="stock.cargo">
|
<button
|
||||||
{{ stock.cargo.id }}
|
class="btn btn--image"
|
||||||
</span>
|
:data-disabled="stockIsEmpty"
|
||||||
<span class="stock-info__length"> {{ stock.length }}m </span>
|
:disabled="stockIsEmpty"
|
||||||
<span class="stock-info__mass">{{ stock.cargo ? stock.cargo.totalMass : stock.mass }}t </span>
|
@click="resetStock"
|
||||||
<span class="stock-info__speed"> {{ stock.maxSpeed }}km/h </span>
|
>
|
||||||
</div>
|
<img src="/images/icon-reset.svg" alt="reset icon" />
|
||||||
</li>
|
{{ $t('stocklist.action-reset') }}
|
||||||
</TransitionGroup>
|
</button>
|
||||||
</ul>
|
|
||||||
|
<button
|
||||||
|
class="btn btn--image"
|
||||||
|
:data-disabled="stockIsEmpty"
|
||||||
|
:disabled="stockIsEmpty"
|
||||||
|
@click="shuffleCars"
|
||||||
|
>
|
||||||
|
<img src="/images/icon-shuffle.svg" alt="shuffle icon" />
|
||||||
|
{{ $t('stocklist.action-shuffle') }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="stock_controls" :data-disabled="store.chosenStockListIndex == -1">
|
||||||
|
<button
|
||||||
|
class="btn btn--image"
|
||||||
|
:tabindex="store.chosenStockListIndex == -1 ? -1 : 0"
|
||||||
|
@click="moveUpStock(store.chosenStockListIndex)"
|
||||||
|
>
|
||||||
|
<img :src="getIconURL('higher')" alt="move up vehicle" />
|
||||||
|
{{ $t('stocklist.action-move-up') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
class="btn btn--image"
|
||||||
|
:tabindex="store.chosenStockListIndex == -1 ? -1 : 0"
|
||||||
|
@click="moveDownStock(store.chosenStockListIndex)"
|
||||||
|
>
|
||||||
|
<img :src="getIconURL('lower')" alt="move down vehicle" />
|
||||||
|
{{ $t('stocklist.action-move-down') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button
|
||||||
|
class="btn btn--image"
|
||||||
|
:tabindex="store.chosenStockListIndex == -1 ? -1 : 0"
|
||||||
|
@click="removeStock(store.chosenStockListIndex)"
|
||||||
|
>
|
||||||
|
<img :src="getIconURL('remove')" alt="remove vehicle" />
|
||||||
|
{{ $t('stocklist.action-remove') }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="stock_specs">
|
||||||
|
<b class="real-stock-info" v-if="chosenRealComposition">
|
||||||
|
<span class="text--accent">
|
||||||
|
<img :src="getIconURL(chosenRealComposition.type)" :alt="chosenRealComposition.type" />
|
||||||
|
{{ chosenRealComposition.number }} {{ chosenRealComposition.name }}
|
||||||
|
</span>
|
||||||
|
|
|
||||||
|
</b>
|
||||||
|
|
||||||
|
<span>
|
||||||
|
{{ $t('stocklist.mass') }}
|
||||||
|
<span class="text--accent">{{ (store.totalWeight / 1000).toFixed(1) }}t</span>
|
||||||
|
({{ $t('stocklist.mass-accepted') }}:
|
||||||
|
<span class="text--accent">{{
|
||||||
|
store.acceptableWeight ? `${~~(store.acceptableWeight / 1000)}t` : '-'
|
||||||
|
}}</span
|
||||||
|
>) - {{ $t('stocklist.length') }}:
|
||||||
|
<span class="text--accent">{{ store.totalLength }}m</span>
|
||||||
|
- {{ $t('stocklist.vmax') }}
|
||||||
|
<span tabindex="0" :data-tooltip="$t('stocklist.disclaimer')">(?)</span>:
|
||||||
|
<span class="text--accent">{{ store.maxStockSpeed }} km/h</span>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div></div>
|
||||||
|
|
||||||
|
<div class="stock_spawn-settings">
|
||||||
|
<Checkbox :disabled="!store.stockSupportsColdStart" v-model="store.isColdStart">
|
||||||
|
{{ $t('stocklist.coldstart-info') }}
|
||||||
|
</Checkbox>
|
||||||
|
|
||||||
|
<Checkbox :disabled="!store.stockSupportsDoubleManning" v-model="store.isDoubleManned">
|
||||||
|
{{ $t('stocklist.doublemanning-info') }}
|
||||||
|
</Checkbox>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="stock_warnings" v-if="hasAnyWarnings">
|
||||||
|
<div class="warning" v-if="locoNotSuitable">
|
||||||
|
(!) {{ $t('stocklist.warning-not-suitable') }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="warning" v-if="lengthExceeded && store.isTrainPassenger">
|
||||||
|
(!) {{ $t('stocklist.warning-passenger-too-long') }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="warning" v-if="lengthExceeded && !store.isTrainPassenger">
|
||||||
|
(!) {{ $t('stocklist.warning-freight-too-long') }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="warning" v-if="teamOnlyVehicles.length > 0">
|
||||||
|
(!)
|
||||||
|
{{
|
||||||
|
$t('stocklist.warning-team-only-vehicle', [
|
||||||
|
teamOnlyVehicles.map((v) => v.vehicleRef.type).join(', '),
|
||||||
|
])
|
||||||
|
}}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="warning" v-if="weightExceeded">
|
||||||
|
(!)
|
||||||
|
<i18n-t keypath="stocklist.warning-too-heavy">
|
||||||
|
<template #href>
|
||||||
|
<a
|
||||||
|
target="_blank"
|
||||||
|
href="https://docs.google.com/spreadsheets/d/1KVa5vn2d8XGkXQFwbavVudwKqUQxbLOucHWs2VYqAUE"
|
||||||
|
>
|
||||||
|
{{ $t('stocklist.acceptable-mass-docs') }}
|
||||||
|
</a>
|
||||||
|
</template>
|
||||||
|
</i18n-t>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="warning" v-if="locoCountExceeded">
|
||||||
|
{{ $t('stocklist.warning-too-many-locos') }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<StockThumbnails :onListItemClick="onListItemClick" />
|
||||||
|
|
||||||
|
<!-- Stock list -->
|
||||||
|
<div class="list-wrapper">
|
||||||
|
<div v-if="stockIsEmpty" class="list-empty">
|
||||||
|
<div class="stock-info">{{ $t('stocklist.list-empty') }}</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ul v-else>
|
||||||
|
<transition-group name="stock-list-anim">
|
||||||
|
<li
|
||||||
|
v-for="(stock, i) in store.stockList"
|
||||||
|
:key="stock.id"
|
||||||
|
:class="{ loco: isTractionUnit(stock.vehicleRef) }"
|
||||||
|
tabindex="0"
|
||||||
|
@click="onListItemClick(i)"
|
||||||
|
@keydown.enter="onListItemClick(i)"
|
||||||
|
@keydown.w="moveUpStock(i)"
|
||||||
|
@keydown.s="moveDownStock(i)"
|
||||||
|
@keydown.backspace="removeStock(i)"
|
||||||
|
ref="itemRefs"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="stock-info"
|
||||||
|
@dragstart="onDragStart(i)"
|
||||||
|
@drop="onDrop($event, i)"
|
||||||
|
@dragover="allowDrop"
|
||||||
|
draggable="true"
|
||||||
|
>
|
||||||
|
<span class="stock-info-no" :data-selected="i == store.chosenStockListIndex">
|
||||||
|
<span v-if="i == store.chosenStockListIndex">• </span>
|
||||||
|
{{ i + 1 }}.
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span
|
||||||
|
class="stock-info-type"
|
||||||
|
:data-sponsor-only="
|
||||||
|
stock.vehicleRef.sponsorOnlyTimestamp &&
|
||||||
|
stock.vehicleRef.sponsorOnlyTimestamp > Date.now()
|
||||||
|
"
|
||||||
|
:data-team-only="stock.vehicleRef.teamOnly"
|
||||||
|
>
|
||||||
|
{{
|
||||||
|
isTractionUnit(stock.vehicleRef)
|
||||||
|
? stock.vehicleRef.type
|
||||||
|
: getCarSpecFromType(stock.vehicleRef.type)
|
||||||
|
}}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span class="stock-info-cargo" v-if="stock.cargo">
|
||||||
|
{{ stock.cargo.id }}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span class="stock-info-length">{{ stock.vehicleRef.length }}m</span>
|
||||||
|
|
||||||
|
<span class="stock-info-mass">
|
||||||
|
{{ ((stock.vehicleRef.weight + (stock.cargo?.weight ?? 0)) / 1000).toFixed(1) }}t
|
||||||
|
</span>
|
||||||
|
<span class="stock-info-speed">{{ stock.vehicleRef.maxSpeed }}km/h</span>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
</transition-group>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</section>
|
</section>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -163,17 +232,18 @@ import { defineComponent } from 'vue';
|
|||||||
|
|
||||||
import { useStore } from '../../store';
|
import { useStore } from '../../store';
|
||||||
|
|
||||||
import warningsMixin from '../../mixins/warningsMixin';
|
|
||||||
import imageMixin from '../../mixins/imageMixin';
|
import imageMixin from '../../mixins/imageMixin';
|
||||||
import stockPreviewMixin from '../../mixins/stockPreviewMixin';
|
import stockPreviewMixin from '../../mixins/stockPreviewMixin';
|
||||||
import StockThumbnails from '../utils/StockThumbnails.vue';
|
import StockThumbnails from '../utils/StockThumbnails.vue';
|
||||||
import stockMixin from '../../mixins/stockMixin';
|
import stockMixin from '../../mixins/stockMixin';
|
||||||
|
import Checkbox from '../common/Checkbox.vue';
|
||||||
|
import { isTractionUnit } from '../../utils/vehicleUtils';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: 'stock-list',
|
name: 'stock-list',
|
||||||
components: { StockThumbnails },
|
components: { StockThumbnails, Checkbox },
|
||||||
|
|
||||||
mixins: [warningsMixin, imageMixin, stockMixin, stockPreviewMixin],
|
mixins: [imageMixin, stockMixin, stockPreviewMixin],
|
||||||
|
|
||||||
setup() {
|
setup() {
|
||||||
const store = useStore();
|
const store = useStore();
|
||||||
@@ -191,15 +261,25 @@ export default defineComponent({
|
|||||||
}),
|
}),
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
|
chosenRealComposition() {
|
||||||
|
const currentStockString = this.store.stockList.map((s) => s.vehicleRef.type).join(';');
|
||||||
|
|
||||||
|
return this.store.realCompositionList.find((rc) => rc.stockString == currentStockString);
|
||||||
|
},
|
||||||
|
|
||||||
stockString() {
|
stockString() {
|
||||||
if (this.store.stockList.length == 0) return '';
|
if (this.store.stockList.length == 0) return '';
|
||||||
|
|
||||||
const includeColdStart = this.store.isColdStart && this.store.stockSupportsColdStart;
|
const includeColdStart = this.store.isColdStart && this.store.stockSupportsColdStart;
|
||||||
const includeDoubleManned = this.store.isDoubleManned && this.store.stockSupportsDoubleManning;
|
const includeDoubleManned =
|
||||||
|
this.store.isDoubleManned && this.store.stockSupportsDoubleManning;
|
||||||
|
|
||||||
return this.store.stockList
|
return this.store.stockList
|
||||||
.map((stock, i) => {
|
.map((stock, i) => {
|
||||||
let stockTypeStr = stock.isLoco || !stock.cargo ? stock.type : `${stock.type}:${stock.cargo.id}`;
|
let stockTypeStr =
|
||||||
|
isTractionUnit(stock.vehicleRef) || !stock.cargo
|
||||||
|
? stock.vehicleRef.type
|
||||||
|
: `${stock.vehicleRef.type}:${stock.cargo.id}`;
|
||||||
|
|
||||||
if (i == 0 && (includeColdStart || includeDoubleManned))
|
if (i == 0 && (includeColdStart || includeDoubleManned))
|
||||||
return `${stockTypeStr},${includeColdStart ? 'c' : ''}${includeDoubleManned ? 'd' : ''}`;
|
return `${stockTypeStr},${includeColdStart ? 'c' : ''}${includeDoubleManned ? 'd' : ''}`;
|
||||||
@@ -214,15 +294,60 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
chosenStockVehicle() {
|
chosenStockVehicle() {
|
||||||
return this.store.chosenStockListIndex == -1 ? undefined : this.store.stockList[this.store.chosenStockListIndex];
|
return this.store.chosenStockListIndex == -1
|
||||||
|
? undefined
|
||||||
|
: this.store.stockList[this.store.chosenStockListIndex];
|
||||||
},
|
},
|
||||||
|
|
||||||
stockHasWarnings() {
|
lengthExceeded() {
|
||||||
return this.tooManyLocomotives || this.trainTooHeavy || this.trainTooLong || this.locoNotSuitable;
|
return (
|
||||||
|
(this.store.totalLength > 350 && this.store.isTrainPassenger) ||
|
||||||
|
(this.store.totalLength > 650 && !this.store.isTrainPassenger)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
weightExceeded() {
|
||||||
|
return this.store.acceptableWeight && this.store.totalWeight > this.store.acceptableWeight;
|
||||||
|
},
|
||||||
|
|
||||||
|
locoNotSuitable() {
|
||||||
|
return (
|
||||||
|
!this.store.isTrainPassenger &&
|
||||||
|
this.store.stockList.length > 1 &&
|
||||||
|
!this.store.stockList.every((stock) => isTractionUnit(stock.vehicleRef)) &&
|
||||||
|
this.store.stockList.some(
|
||||||
|
(stock) => isTractionUnit(stock.vehicleRef) && stock.vehicleRef.type.startsWith('EP')
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
locoCountExceeded() {
|
||||||
|
return (
|
||||||
|
this.store.stockList.reduce((acc, stock) => {
|
||||||
|
if (isTractionUnit(stock.vehicleRef)) acc += 1;
|
||||||
|
return acc;
|
||||||
|
}, 0) > 2
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
teamOnlyVehicles() {
|
||||||
|
return this.store.stockList.filter((stock) => stock.vehicleRef.teamOnly);
|
||||||
|
},
|
||||||
|
|
||||||
|
hasAnyWarnings() {
|
||||||
|
return (
|
||||||
|
this.locoCountExceeded ||
|
||||||
|
this.weightExceeded ||
|
||||||
|
this.lengthExceeded ||
|
||||||
|
this.locoNotSuitable ||
|
||||||
|
this.teamOnlyVehicles
|
||||||
|
);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
|
isTractionUnit,
|
||||||
|
|
||||||
copyToClipboard() {
|
copyToClipboard() {
|
||||||
navigator.clipboard.writeText(this.stockString);
|
navigator.clipboard.writeText(this.stockString);
|
||||||
|
|
||||||
@@ -238,7 +363,11 @@ export default defineComponent({
|
|||||||
onListItemClick(stockID: number) {
|
onListItemClick(stockID: number) {
|
||||||
const stock = this.store.stockList[stockID];
|
const stock = this.store.stockList[stockID];
|
||||||
|
|
||||||
this.store.chosenStockListIndex = this.store.chosenStockListIndex == stockID && this.store.chosenVehicle?.type == stock.type ? -1 : stockID;
|
this.store.chosenStockListIndex =
|
||||||
|
this.store.chosenStockListIndex == stockID &&
|
||||||
|
this.store.chosenVehicle?.type == stock.vehicleRef.type
|
||||||
|
? -1
|
||||||
|
: stockID;
|
||||||
|
|
||||||
if (this.store.chosenStockListIndex == -1) {
|
if (this.store.chosenStockListIndex == -1) {
|
||||||
this.store.chosenVehicle = null;
|
this.store.chosenVehicle = null;
|
||||||
@@ -267,20 +396,6 @@ export default defineComponent({
|
|||||||
this.store.chosenStockListIndex = -1;
|
this.store.chosenStockListIndex = -1;
|
||||||
},
|
},
|
||||||
|
|
||||||
addStock(index: number) {
|
|
||||||
if (index == -1) return;
|
|
||||||
|
|
||||||
this.store.stockList[index].count++;
|
|
||||||
},
|
|
||||||
|
|
||||||
subStock(index: number) {
|
|
||||||
if (index == -1) return;
|
|
||||||
|
|
||||||
if (this.store.stockList[index].count < 2) return;
|
|
||||||
|
|
||||||
this.store.stockList[index].count--;
|
|
||||||
},
|
|
||||||
|
|
||||||
removeStock(index: number) {
|
removeStock(index: number) {
|
||||||
if (index == -1) return;
|
if (index == -1) return;
|
||||||
|
|
||||||
@@ -314,7 +429,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
shuffleCars() {
|
shuffleCars() {
|
||||||
const availableIndexes = this.store.stockList.reduce((acc, stock, i) => {
|
const availableIndexes = this.store.stockList.reduce((acc, stock, i) => {
|
||||||
if (!stock.isLoco) acc.push(i);
|
if (!isTractionUnit(stock.vehicleRef)) acc.push(i);
|
||||||
|
|
||||||
return acc;
|
return acc;
|
||||||
}, [] as number[]);
|
}, [] as number[]);
|
||||||
@@ -324,7 +439,8 @@ export default defineComponent({
|
|||||||
|
|
||||||
availableIndexes.splice(i, -1);
|
availableIndexes.splice(i, -1);
|
||||||
|
|
||||||
const randAvailableIndex = availableIndexes[Math.floor(Math.random() * availableIndexes.length)];
|
const randAvailableIndex =
|
||||||
|
availableIndexes[Math.floor(Math.random() * availableIndexes.length)];
|
||||||
const tempSwap = this.store.stockList[randAvailableIndex];
|
const tempSwap = this.store.stockList[randAvailableIndex];
|
||||||
|
|
||||||
this.store.stockList[randAvailableIndex] = this.store.stockList[i];
|
this.store.stockList[randAvailableIndex] = this.store.stockList[i];
|
||||||
@@ -335,7 +451,7 @@ export default defineComponent({
|
|||||||
downloadStock() {
|
downloadStock() {
|
||||||
if (this.store.stockList.length == 0) return alert(this.$t('stocklist.alert-empty'));
|
if (this.store.stockList.length == 0) return alert(this.$t('stocklist.alert-empty'));
|
||||||
|
|
||||||
const defaultName = `${this.store.chosenRealStockName || this.store.stockList[0].type} ${this.store.totalMass}t; ${
|
const defaultName = `${this.chosenRealComposition ? this.chosenRealComposition.stockId + ' ' : ''}${this.store.stockList[0].vehicleRef.type} ${(this.store.totalWeight / 1000).toFixed(1)}t; ${
|
||||||
this.store.totalLength
|
this.store.totalLength
|
||||||
}m; vmax ${this.store.maxStockSpeed}`;
|
}m; vmax ${this.store.maxStockSpeed}`;
|
||||||
|
|
||||||
@@ -373,7 +489,7 @@ export default defineComponent({
|
|||||||
this.loadStockFromString(stockString);
|
this.loadStockFromString(stockString);
|
||||||
};
|
};
|
||||||
|
|
||||||
reader.onerror = (err) => console.log(err);
|
reader.onerror = (err) => console.error(err);
|
||||||
|
|
||||||
inputEl.value = '';
|
inputEl.value = '';
|
||||||
},
|
},
|
||||||
@@ -408,13 +524,15 @@ export default defineComponent({
|
|||||||
@import '../../styles/global';
|
@import '../../styles/global';
|
||||||
@import '../../styles/tab.scss';
|
@import '../../styles/tab.scss';
|
||||||
|
|
||||||
.stock-list-tab {
|
.tab_content {
|
||||||
display: grid;
|
display: flex;
|
||||||
grid-gap: 0.5em;
|
flex-direction: column;
|
||||||
|
gap: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.warning {
|
.warning {
|
||||||
padding: 0.25em;
|
padding: 0.25em;
|
||||||
|
margin: 0.25em 0;
|
||||||
background: $accentColor;
|
background: $accentColor;
|
||||||
color: black;
|
color: black;
|
||||||
|
|
||||||
@@ -430,9 +548,9 @@ export default defineComponent({
|
|||||||
display: flex;
|
display: flex;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
flex-wrap: wrap;
|
|
||||||
|
|
||||||
padding: 0.5em;
|
padding: 0.5em;
|
||||||
|
|
||||||
@@ -447,21 +565,6 @@ export default defineComponent({
|
|||||||
|
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
input#stock-count {
|
|
||||||
width: 3em;
|
|
||||||
|
|
||||||
margin: 0;
|
|
||||||
padding: 0.25em;
|
|
||||||
outline: none;
|
|
||||||
border: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
button {
|
|
||||||
img {
|
|
||||||
margin-right: 0.25em;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.stock_actions {
|
.stock_actions {
|
||||||
@@ -484,39 +587,6 @@ export default defineComponent({
|
|||||||
.stock_spawn-settings {
|
.stock_spawn-settings {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
|
|
||||||
label > input {
|
|
||||||
position: absolute;
|
|
||||||
clip: rect(1px, 1px, 1px, 1px);
|
|
||||||
padding: 0;
|
|
||||||
border: 0;
|
|
||||||
height: 1px;
|
|
||||||
width: 1px;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
label {
|
|
||||||
padding: 0.25em 0.5em;
|
|
||||||
border-radius: 0.25em;
|
|
||||||
background-color: #222;
|
|
||||||
color: #aaa;
|
|
||||||
cursor: pointer;
|
|
||||||
|
|
||||||
text-transform: uppercase;
|
|
||||||
transition: color 200ms;
|
|
||||||
|
|
||||||
&::before {
|
|
||||||
content: '\2716';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
label[data-checked='true'] {
|
|
||||||
color: palegreen;
|
|
||||||
|
|
||||||
&::before {
|
|
||||||
content: '\2714';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.real-stock-info {
|
.real-stock-info {
|
||||||
@@ -525,9 +595,19 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ul {
|
.list-wrapper {
|
||||||
position: relative;
|
position: relative;
|
||||||
overflow: auto;
|
}
|
||||||
|
|
||||||
|
.list-empty {
|
||||||
|
background-color: $secondaryColor;
|
||||||
|
border-radius: 0.5em;
|
||||||
|
padding: 0.75em;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul {
|
||||||
|
overflow-y: scroll;
|
||||||
height: 500px;
|
height: 500px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -545,16 +625,11 @@ ul > li {
|
|||||||
&:focus-visible {
|
&:focus-visible {
|
||||||
outline: 1px solid white;
|
outline: 1px solid white;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.list-empty {
|
|
||||||
background-color: $secondaryColor;
|
|
||||||
border-radius: 0.5em;
|
|
||||||
padding: 0.75em;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
li > .stock-info {
|
li > .stock-info {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
gap: 0.25em;
|
||||||
|
|
||||||
color: white;
|
color: white;
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
@@ -563,46 +638,39 @@ li > .stock-info {
|
|||||||
|
|
||||||
& > span {
|
& > span {
|
||||||
padding: 0.5em;
|
padding: 0.5em;
|
||||||
margin-right: 0.25em;
|
|
||||||
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.sponsor {
|
.stock-info-no,
|
||||||
color: salmon;
|
.stock-info-type {
|
||||||
|
background-color: $secondaryColor;
|
||||||
|
|
||||||
|
&[data-team-only='true'] {
|
||||||
|
color: $teamColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
&[data-sponsor-only='true'] {
|
||||||
|
color: $sponsorColor;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.stock-info {
|
.stock-info-no {
|
||||||
&__no,
|
min-width: 3.5em;
|
||||||
&__type {
|
text-align: right;
|
||||||
background-color: $secondaryColor;
|
|
||||||
}
|
|
||||||
|
|
||||||
&__count {
|
&[data-selected='true'] {
|
||||||
background-color: #e04e3e;
|
color: $accentColor;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
&__no {
|
.stock-info-cargo {
|
||||||
min-width: 3.5em;
|
background-color: #333;
|
||||||
text-align: right;
|
}
|
||||||
|
|
||||||
&[data-selected='true'] {
|
.stock-info-length,
|
||||||
color: $accentColor;
|
.stock-info-mass,
|
||||||
}
|
.stock-info-speed {
|
||||||
}
|
background-color: #555;
|
||||||
|
|
||||||
&__cargo {
|
|
||||||
background-color: #333;
|
|
||||||
}
|
|
||||||
|
|
||||||
&__length,
|
|
||||||
&__mass,
|
|
||||||
&__speed {
|
|
||||||
background-color: #555;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.stock-list-anim {
|
.stock-list-anim {
|
||||||
|
|||||||
@@ -5,75 +5,80 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="tab_content">
|
<div class="tab_content">
|
||||||
<div class="actions-panel">
|
<div class="actions">
|
||||||
<div class="actions-panel_vehicles">
|
<label>
|
||||||
<button class="btn" :data-chosen="currentFilterMode == 'tractions'" @click="toggleFilter('tractions')">
|
<span>{{ $t('wiki.labels.search-vehicle') }}</span>
|
||||||
{{ $t('wiki.action-vehicles') }}
|
<input
|
||||||
</button>
|
type="text"
|
||||||
<button class="btn" :data-chosen="currentFilterMode == 'carriages'" @click="toggleFilter('carriages')">
|
:placeholder="$t('wiki.labels.search-vehicle-placeholder')"
|
||||||
{{ $t('wiki.action-carriages') }}
|
v-model="searchedVehicleTypeName"
|
||||||
</button>
|
/>
|
||||||
</div>
|
</label>
|
||||||
|
|
||||||
<div class="actions-panel_search">
|
<label>
|
||||||
<input type="text" :placeholder="$t('wiki.search')" v-model="searchedVehicleTypeName" />
|
<span>{{ $t('wiki.labels.vehicles') }}</span>
|
||||||
</div>
|
<select name="filter-type" id="filter-type" v-model="filterType">
|
||||||
|
<option v-for="filter in filters" :key="filter" :value="filter">
|
||||||
|
{{ $t(`wiki.filters.${filter}`) }}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<label>
|
||||||
|
<span>{{ $t('wiki.labels.sort-by') }}</span>
|
||||||
|
<select name="sorter-type" id="sorter-type" v-model="sorterType">
|
||||||
|
<option v-for="sorter in sorters" :key="sorter" :value="sorter">
|
||||||
|
{{ $t(`wiki.sort-by.${sorter}`) }}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<label>
|
||||||
|
<span>{{ $t('wiki.labels.sort-direction') }}</span>
|
||||||
|
|
||||||
|
<select name="sorter-direction" id="sorter-direction" v-model="sorterDirection">
|
||||||
|
<option value="asc">{{ $t('wiki.sort-direction.asc') }}</option>
|
||||||
|
<option value="desc">{{ $t('wiki.sort-direction.desc') }}</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="table-wrapper" ref="table-wrapper">
|
<ul class="vehicles" ref="vehicles">
|
||||||
<table>
|
<li
|
||||||
<thead>
|
v-for="vehicle in computedVehicles"
|
||||||
<tr>
|
:key="vehicle.type"
|
||||||
<th v-for="header in visibleHeaders" @click="toggleSorter(header)" :key="header.id">
|
:data-preview="vehicle.type === store.chosenVehicle?.type"
|
||||||
{{ $t(`wiki.header.${header.id}`) }}
|
@click="previewVehicle(vehicle)"
|
||||||
|
@dblclick="addVehicle(vehicle)"
|
||||||
|
@keydown.enter="onVehicleSelect(vehicle)"
|
||||||
|
tabindex="0"
|
||||||
|
>
|
||||||
|
<img loading="lazy" width="120" :src="getThumbnailURL(vehicle.type, 'small')" />
|
||||||
|
|
||||||
<span v-if="currentSorter.id == header.id">
|
<span>
|
||||||
{{ currentSorter.direction == 1 ? `⇑` : `⇓` }}
|
<span
|
||||||
</span>
|
class="vehicle-name"
|
||||||
</th>
|
:class="{
|
||||||
</tr>
|
'sponsor-only':
|
||||||
</thead>
|
vehicle.sponsorOnlyTimestamp && vehicle.sponsorOnlyTimestamp > Date.now(),
|
||||||
|
'team-only': vehicle.teamOnly,
|
||||||
<tbody>
|
}"
|
||||||
<tr
|
|
||||||
v-for="{ vehicle, show } in computedTableData"
|
|
||||||
tabindex="0"
|
|
||||||
v-show="show"
|
|
||||||
:key="vehicle.type"
|
|
||||||
@click="previewVehicle(vehicle)"
|
|
||||||
@keydown.enter="previewVehicle(vehicle)"
|
|
||||||
@dblclick="addVehicle(vehicle)"
|
|
||||||
>
|
>
|
||||||
<td style="width: 120px">
|
<b>{{ vehicle.type.replace(/_/g, ' ') }}</b>
|
||||||
<img
|
</span>
|
||||||
width="120"
|
|
||||||
:src="getThumbnailURL(vehicle.type, 'small')"
|
|
||||||
:alt="`${vehicle.type}`"
|
|
||||||
loading="lazy"
|
|
||||||
@error="(e) => ((e.target as HTMLElement).style.display = 'none')"
|
|
||||||
/>
|
|
||||||
</td>
|
|
||||||
|
|
||||||
<td :data-sponsoronly="vehicle.isSponsorsOnly">{{ vehicle.type }}</td>
|
<div class="vehicle-group">
|
||||||
|
{{ $t(`wiki.${vehicle.group}`) }} |
|
||||||
|
{{ isTractionUnit(vehicle) ? vehicle.cabinType : vehicle.constructionType }}
|
||||||
|
</div>
|
||||||
|
|
||||||
<td v-if="isLocomotive(vehicle)">{{ $t(`wiki.${vehicle.power}`) }}</td>
|
<div class="vehicle-props">
|
||||||
<td v-else>{{ $t(`wiki.${vehicle.useType}`) }}</td>
|
{{ vehicle.length }}m | {{ (vehicle.weight / 1000).toFixed(1) }}t |
|
||||||
|
{{ vehicle.maxSpeed }}km/h
|
||||||
<td>{{ vehicle.constructionType }}</td>
|
</div>
|
||||||
<td>{{ vehicle.length }}m</td>
|
</span>
|
||||||
<td>{{ vehicle.mass }}t</td>
|
</li>
|
||||||
<td>{{ vehicle.maxSpeed }}km/h</td>
|
</ul>
|
||||||
|
|
||||||
<td v-if="currentFilterMode == 'carriages'">{{ !isLocomotive(vehicle) ? vehicle.cargoList.length : '---' }}</td>
|
|
||||||
<td v-if="currentFilterMode == 'tractions'">
|
|
||||||
{{ isLocomotive(vehicle) ? (vehicle.coldStart ? `✓` : '✗') : '---' }}
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
|
|
||||||
<span ref="table-bottom"></span>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
</template>
|
</template>
|
||||||
@@ -82,35 +87,18 @@
|
|||||||
import { defineComponent } from 'vue';
|
import { defineComponent } from 'vue';
|
||||||
import { useStore } from '../../store';
|
import { useStore } from '../../store';
|
||||||
import stockPreviewMixin from '../../mixins/stockPreviewMixin';
|
import stockPreviewMixin from '../../mixins/stockPreviewMixin';
|
||||||
import { Vehicle } from '../../types';
|
import { IVehicle } from '../../types';
|
||||||
import { isLocomotive } from '../../utils/vehicleUtils';
|
import { isTractionUnit } from '../../utils/vehicleUtils';
|
||||||
import stockMixin from '../../mixins/stockMixin';
|
import stockMixin from '../../mixins/stockMixin';
|
||||||
import imageMixin from '../../mixins/imageMixin';
|
import imageMixin from '../../mixins/imageMixin';
|
||||||
|
|
||||||
type SorterID = 'type' | 'constructionType' | 'image' | 'length' | 'mass' | 'maxSpeed' | 'cargoCount' | 'group' | 'coldStart';
|
const sorters = ['type', 'group', 'length', 'weight', 'maxSpeed'] as const;
|
||||||
|
const filters = ['vehicles-all', 'vehicles-traction', 'vehicles-wagon'] as const;
|
||||||
|
|
||||||
interface IWikiHeader {
|
type SorterType = (typeof sorters)[number];
|
||||||
id: SorterID;
|
type SorterDirection = 'asc' | 'desc';
|
||||||
sortable: boolean;
|
|
||||||
for: 'all' | 'carriages' | 'tractions';
|
|
||||||
}
|
|
||||||
|
|
||||||
interface IWikiRow {
|
type FilterType = (typeof filters)[number];
|
||||||
vehicle: Vehicle;
|
|
||||||
show: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
const headers: IWikiHeader[] = [
|
|
||||||
{ id: 'image', sortable: false, for: 'all' },
|
|
||||||
{ id: 'type', sortable: true, for: 'all' },
|
|
||||||
{ id: 'group', sortable: true, for: 'all' },
|
|
||||||
{ id: 'constructionType', sortable: true, for: 'all' },
|
|
||||||
{ id: 'length', sortable: true, for: 'all' },
|
|
||||||
{ id: 'mass', sortable: true, for: 'all' },
|
|
||||||
{ id: 'maxSpeed', sortable: true, for: 'all' },
|
|
||||||
{ id: 'coldStart', sortable: true, for: 'tractions' },
|
|
||||||
{ id: 'cargoCount', sortable: true, for: 'carriages' },
|
|
||||||
];
|
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
mixins: [stockPreviewMixin, stockMixin, imageMixin],
|
mixins: [stockPreviewMixin, stockMixin, imageMixin],
|
||||||
@@ -118,105 +106,104 @@ export default defineComponent({
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
store: useStore(),
|
store: useStore(),
|
||||||
headers,
|
observer: null as IntersectionObserver | null,
|
||||||
|
|
||||||
scrollTop: 0,
|
sorters: sorters,
|
||||||
|
filters: filters,
|
||||||
|
|
||||||
searchedVehicleTypeName: '',
|
searchedVehicleTypeName: '',
|
||||||
|
|
||||||
currentSorter: {
|
sorterType: 'type' as SorterType,
|
||||||
id: 'type' as SorterID,
|
sorterDirection: 'asc' as SorterDirection,
|
||||||
direction: 1,
|
|
||||||
},
|
|
||||||
|
|
||||||
currentFilterMode: 'all' as 'all' | 'tractions' | 'carriages',
|
filterType: 'vehicles-all' as FilterType,
|
||||||
|
|
||||||
|
lastScrollTop: 0,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
activated() {
|
deactivated() {
|
||||||
const tableWrapperRef = this.$refs['table-wrapper'] as HTMLElement;
|
this.lastScrollTop = (this.$refs['vehicles'] as HTMLUListElement)?.scrollTop || 0;
|
||||||
|
},
|
||||||
|
|
||||||
tableWrapperRef.scrollTo({
|
activated() {
|
||||||
top: this.scrollTop,
|
(this.$refs['vehicles'] as HTMLUListElement)?.scrollTo({ top: this.lastScrollTop });
|
||||||
});
|
},
|
||||||
|
|
||||||
|
watch: {
|
||||||
|
computedVehicles() {
|
||||||
|
const vehiclesRef = this.$refs['vehicles'] as HTMLElement;
|
||||||
|
|
||||||
|
vehiclesRef.scrollTo({
|
||||||
|
top: 0,
|
||||||
|
});
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
isLocomotive,
|
isTractionUnit,
|
||||||
|
|
||||||
toggleFilter(name: typeof this.currentFilterMode) {
|
onVehicleSelect(vehicle: IVehicle) {
|
||||||
this.currentFilterMode = this.currentFilterMode == name ? 'all' : name;
|
if (this.store.chosenVehicle?.type === vehicle.type) this.addVehicle(vehicle);
|
||||||
|
this.previewVehicle(vehicle);
|
||||||
},
|
},
|
||||||
|
|
||||||
toggleSorter(header: IWikiHeader) {
|
filterVehicles(v: IVehicle) {
|
||||||
if (!header.sortable) return;
|
if (this.searchedVehicleTypeName)
|
||||||
|
return v.type
|
||||||
|
.toLocaleLowerCase()
|
||||||
|
.includes(this.searchedVehicleTypeName.toLocaleLowerCase());
|
||||||
|
|
||||||
if (header.id == this.currentSorter.id) this.currentSorter.direction *= -1;
|
switch (this.filterType) {
|
||||||
this.currentSorter.id = header.id;
|
case 'vehicles-all':
|
||||||
},
|
return true;
|
||||||
|
case 'vehicles-traction':
|
||||||
sortTableRows(row1: IWikiRow, row2: IWikiRow) {
|
return isTractionUnit(v);
|
||||||
if (!row1.show) return 0;
|
case 'vehicles-wagon':
|
||||||
|
return !isTractionUnit(v);
|
||||||
const { id, direction } = this.currentSorter;
|
|
||||||
|
|
||||||
switch (id) {
|
|
||||||
case 'type':
|
|
||||||
case 'constructionType':
|
|
||||||
case 'group':
|
|
||||||
return direction == 1 ? row1.vehicle[id].localeCompare(row2.vehicle[id]) : row2.vehicle[id].localeCompare(row1.vehicle[id]);
|
|
||||||
|
|
||||||
case 'mass':
|
|
||||||
case 'length':
|
|
||||||
case 'maxSpeed':
|
|
||||||
return Math.sign(row1.vehicle[id] - row2.vehicle[id]) * direction;
|
|
||||||
|
|
||||||
case 'cargoCount':
|
|
||||||
return (
|
|
||||||
(!isLocomotive(row1.vehicle) ? Math.sign(row1.vehicle.cargoList.length || -1) : -1) -
|
|
||||||
(!isLocomotive(row2.vehicle) ? (row2.vehicle.cargoList.length || -1) * direction : -1)
|
|
||||||
);
|
|
||||||
|
|
||||||
case 'coldStart':
|
|
||||||
return (
|
|
||||||
((isLocomotive(row1.vehicle) && row1.vehicle.coldStart ? 1 : -1) - (isLocomotive(row2.vehicle) && row2.vehicle.coldStart ? 1 : -1)) *
|
|
||||||
direction
|
|
||||||
);
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
break;
|
return false;
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
|
||||||
return direction == 1 ? row1.vehicle.type.localeCompare(row2.vehicle.type) : row2.vehicle.type.localeCompare(row1.vehicle.type);
|
sortVehicles(v1: IVehicle, v2: IVehicle) {
|
||||||
|
const direction = this.sorterDirection == 'asc' ? 1 : -1;
|
||||||
|
|
||||||
|
switch (this.sorterType) {
|
||||||
|
case 'type':
|
||||||
|
case 'group':
|
||||||
|
return direction * v1[this.sorterType].localeCompare(v2[this.sorterType]);
|
||||||
|
|
||||||
|
case 'weight':
|
||||||
|
case 'length':
|
||||||
|
case 'maxSpeed':
|
||||||
|
return Math.sign(v1[this.sorterType] - v2[this.sorterType]) * direction;
|
||||||
|
|
||||||
|
// case 'cargoCount':
|
||||||
|
// return (
|
||||||
|
// Math.sign(
|
||||||
|
// (!isTractionUnit(v1) ? v1.cargoTypes.length || -1 : -1) -
|
||||||
|
// (!isTractionUnit(row2.vehicle) ? row2.vehicle.cargoTypes.length || -1 : -1)
|
||||||
|
// ) * direction
|
||||||
|
// );
|
||||||
|
|
||||||
|
// case 'coldStart':
|
||||||
|
// return (
|
||||||
|
// ((isTractionUnit(v1) && v1.coldStart ? 1 : -1) -
|
||||||
|
// (isTractionUnit(row2.vehicle) && row2.vehicle.coldStart ? 1 : -1)) *
|
||||||
|
// direction
|
||||||
|
// );
|
||||||
|
|
||||||
|
default:
|
||||||
|
return v1.type.localeCompare(v2.type) * direction;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
computedTableData(): IWikiRow[] {
|
computedVehicles() {
|
||||||
return this.store.vehicleDataList
|
return this.store.vehicleDataList.filter(this.filterVehicles).sort(this.sortVehicles);
|
||||||
.map((vehicle) => ({
|
|
||||||
vehicle,
|
|
||||||
show:
|
|
||||||
new RegExp(`${this.searchedVehicleTypeName.trim()}`, 'i').test(vehicle.type) &&
|
|
||||||
(this.currentFilterMode == 'all' ||
|
|
||||||
(this.currentFilterMode == 'tractions' && isLocomotive(vehicle)) ||
|
|
||||||
(this.currentFilterMode == 'carriages' && !isLocomotive(vehicle))),
|
|
||||||
}))
|
|
||||||
.sort((a, b) => this.sortTableRows(a, b));
|
|
||||||
},
|
|
||||||
|
|
||||||
visibleHeaders() {
|
|
||||||
const filtersActive = this.currentFilterMode;
|
|
||||||
|
|
||||||
return this.headers.filter((header) => header.for == 'all' || header.for == filtersActive);
|
|
||||||
},
|
|
||||||
|
|
||||||
areTractionVehiclesShown() {
|
|
||||||
return this.currentFilterMode == 'all' || this.currentFilterMode == 'tractions';
|
|
||||||
},
|
|
||||||
|
|
||||||
areCarriagesShown() {
|
|
||||||
return this.currentFilterMode == 'all' || this.currentFilterMode == 'carriages';
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
@@ -225,90 +212,84 @@ export default defineComponent({
|
|||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/tab.scss';
|
@import '../../styles/tab.scss';
|
||||||
|
|
||||||
.actions-panel {
|
.actions {
|
||||||
display: flex;
|
display: grid;
|
||||||
justify-content: space-between;
|
grid-template-columns: repeat(auto-fit, minmax(10em, 1fr));
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
|
|
||||||
margin: 0.5em 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.actions-panel_vehicles {
|
.actions > label {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.25em;
|
||||||
|
|
||||||
|
span {
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.vehicles {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(320px, 1fr));
|
||||||
|
gap: 0.5em;
|
||||||
|
overflow: auto;
|
||||||
|
|
||||||
|
max-height: 730px;
|
||||||
|
|
||||||
|
margin-top: 0.75em;
|
||||||
|
padding: 0.25em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vehicles > li {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
}
|
|
||||||
|
|
||||||
.actions-panel_search {
|
background-color: #161c2e;
|
||||||
input {
|
padding: 0.5em;
|
||||||
width: auto;
|
|
||||||
|
min-height: 75px;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
&[data-preview='true'] {
|
||||||
|
background-color: #435288;
|
||||||
|
}
|
||||||
|
|
||||||
|
& > span {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: space-between;
|
||||||
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.table-wrapper {
|
.vehicle-name {
|
||||||
overflow: auto;
|
overflow: hidden;
|
||||||
height: 750px;
|
text-wrap: nowrap;
|
||||||
max-height: 95vh;
|
text-overflow: ellipsis;
|
||||||
}
|
}
|
||||||
|
|
||||||
.wiki-list table {
|
.sponsor-only {
|
||||||
border-collapse: collapse;
|
color: $sponsorColor;
|
||||||
width: 100%;
|
|
||||||
|
|
||||||
thead {
|
&::after {
|
||||||
position: sticky;
|
content: '*';
|
||||||
top: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
th {
|
|
||||||
background-color: #111;
|
|
||||||
padding: 0.5em;
|
|
||||||
user-select: none;
|
|
||||||
-moz-user-select: none;
|
|
||||||
-webkit-user-select: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
tr {
|
|
||||||
cursor: pointer;
|
|
||||||
background-color: #333;
|
|
||||||
|
|
||||||
&:first-child {
|
|
||||||
min-width: 120px;
|
|
||||||
}
|
|
||||||
|
|
||||||
&:nth-child(odd) {
|
|
||||||
background-color: #444;
|
|
||||||
}
|
|
||||||
|
|
||||||
&:hover {
|
|
||||||
background-color: #666;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
td {
|
|
||||||
text-align: center;
|
|
||||||
height: 70px;
|
|
||||||
|
|
||||||
&[data-sponsoronly='true'] {
|
|
||||||
color: salmon;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (max-width: $breakpointMd) {
|
.team-only {
|
||||||
.wiki-list table {
|
color: $teamColor;
|
||||||
th {
|
|
||||||
min-width: 100px;
|
|
||||||
}
|
|
||||||
|
|
||||||
img {
|
&::after {
|
||||||
max-width: 100px;
|
content: '*';
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.vehicle-props {
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
@media screen and (max-width: $breakpointSm) {
|
@media screen and (max-width: $breakpointSm) {
|
||||||
.actions-panel {
|
.actions-panel {
|
||||||
align-items: stretch;
|
align-items: stretch;
|
||||||
|
|||||||
@@ -10,8 +10,8 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent } from "vue";
|
import { defineComponent } from 'vue';
|
||||||
import { useStore } from "../../store";
|
import { useStore } from '../../store';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
data() {
|
data() {
|
||||||
|
|||||||
@@ -5,7 +5,10 @@
|
|||||||
v-for="(stock, stockIndex) in store.stockList"
|
v-for="(stock, stockIndex) in store.stockList"
|
||||||
:key="stockIndex"
|
:key="stockIndex"
|
||||||
:data-selected="store.chosenStockListIndex == stockIndex"
|
:data-selected="store.chosenStockListIndex == stockIndex"
|
||||||
:data-sponsor="stock.isSponsorsOnly"
|
:data-sponsor-only="
|
||||||
|
stock.vehicleRef.sponsorOnlyTimestamp && stock.vehicleRef.sponsorOnlyTimestamp > Date.now()
|
||||||
|
"
|
||||||
|
:data-team-only="stock.vehicleRef.teamOnly"
|
||||||
draggable="true"
|
draggable="true"
|
||||||
@dragstart="onDragStart(stockIndex)"
|
@dragstart="onDragStart(stockIndex)"
|
||||||
@drop="onDrop($event, stockIndex)"
|
@drop="onDrop($event, stockIndex)"
|
||||||
@@ -13,14 +16,14 @@
|
|||||||
@click="onListItemClick(stockIndex)"
|
@click="onListItemClick(stockIndex)"
|
||||||
>
|
>
|
||||||
<b>
|
<b>
|
||||||
{{ stock.type }}
|
{{ stock.vehicleRef.type }}
|
||||||
</b>
|
</b>
|
||||||
|
|
||||||
<img
|
<img
|
||||||
draggable="false"
|
draggable="false"
|
||||||
:src="`https://rj.td2.info.pl/dist/img/thumbnails/${stock.type}.png`"
|
:src="`https://rj.td2.info.pl/dist/img/thumbnails/${stock.vehicleRef.type}.png`"
|
||||||
:alt="stock.type"
|
:alt="stock.vehicleRef.type"
|
||||||
:title="stock.type"
|
:title="stock.vehicleRef.type"
|
||||||
@error="stockImageError($event, stock)"
|
@error="stockImageError($event, stock)"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
@@ -43,7 +46,7 @@ const onListItemClick = (index: number) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const stockImageError = (e: Event, stock: IStock) => {
|
const stockImageError = (e: Event, stock: IStock) => {
|
||||||
(e.target as HTMLImageElement).src = `images/${stock.useType}-unknown.png`;
|
(e.target as HTMLImageElement).src = `images/${stock.vehicleRef.group}-unknown.png`;
|
||||||
};
|
};
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
@@ -52,11 +55,13 @@ watch(
|
|||||||
if (index < 0) return;
|
if (index < 0) return;
|
||||||
|
|
||||||
nextTick(() => {
|
nextTick(() => {
|
||||||
(thumbnailsRef.value as HTMLElement).querySelector(`div:nth-child(${index + 1})`)?.scrollIntoView({
|
(thumbnailsRef.value as HTMLElement)
|
||||||
block: 'nearest',
|
.querySelector(`div:nth-child(${index + 1})`)
|
||||||
inline: 'start',
|
?.scrollIntoView({
|
||||||
behavior: 'smooth',
|
block: 'nearest',
|
||||||
});
|
inline: 'start',
|
||||||
|
behavior: 'smooth',
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@@ -86,10 +91,13 @@ const allowDrop = (e: DragEvent) => {
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
|
@import '../../styles/global.scss';
|
||||||
|
|
||||||
.stock-thumbnails {
|
.stock-thumbnails {
|
||||||
display: flex;
|
display: flex;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
background-color: #353a57;
|
background-color: #353a57;
|
||||||
|
min-height: 100px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.thumbnail-item {
|
.thumbnail-item {
|
||||||
@@ -119,8 +127,12 @@ const allowDrop = (e: DragEvent) => {
|
|||||||
margin: 0 1em;
|
margin: 0 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
&[data-sponsor='true'] > b {
|
&[data-sponsor-only='true'] > b {
|
||||||
color: salmon;
|
color: $sponsorColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
&[data-team-only='true'] > b {
|
||||||
|
color: $teamColor;
|
||||||
}
|
}
|
||||||
|
|
||||||
img {
|
img {
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
{
|
{
|
||||||
"EU06": [650, 2000],
|
"EU06": [650000, 2000000],
|
||||||
"EU07": [650, 2000],
|
"EU07": [650000, 2000000],
|
||||||
"EU07E": [650, 2000],
|
"4E": [650000, 2000000],
|
||||||
"EP07": [650, 650],
|
"EU07E": [650000, 2000000],
|
||||||
"EP08": [650, 650],
|
"EP07": [650000, 650000],
|
||||||
"EP09": [800, 800],
|
"EP08": [650000, 650000],
|
||||||
"ET41": [700, 4000],
|
"EP09": [800000, 800000],
|
||||||
"SM42": [2400, 2400]
|
"ET41": [700000, 4000000],
|
||||||
|
"SM42": [2400000, 2400000]
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,37 +1,61 @@
|
|||||||
{
|
{
|
||||||
"regionNumbers": {
|
"regionNumbers": {
|
||||||
"Warszawa": 1,
|
"Warszawa (1)": 1,
|
||||||
"Lublin": 2,
|
"Lublin (2)": 2,
|
||||||
"Kraków": 3,
|
"Kraków (3)": 3,
|
||||||
"Sosnowiec": 4,
|
"Sosnowiec (4)": 4,
|
||||||
"Gdańsk": 5,
|
"Gdańsk (5)": 5,
|
||||||
"Wrocław": 6,
|
"Wrocław (6)": 6,
|
||||||
"Poznań": 7,
|
"Poznań (7)": 7,
|
||||||
"Szczecin": 8,
|
"Szczecin (8)": 8,
|
||||||
"Rezerwa": 9
|
"Rezerwa (9)": 9
|
||||||
},
|
},
|
||||||
"sameRegions": {
|
"sameRegions": {
|
||||||
"Losowy": [
|
"Losowy": [
|
||||||
10, 11, 19, 91, 93, 97, 99, 20, 22, 29, 30, 33, 39, 40, 44, 49, 94, 50,
|
10, 11, 19, 91, 93, 97, 99, 20, 22, 29, 30, 33, 39, 40, 44, 49, 94, 50, 55, 59, 90, 95, 96,
|
||||||
55, 59, 90, 95, 96, 66, 60, 69, 77, 70, 79, 88, 80, 89, 92, 98
|
66, 60, 69, 77, 70, 79, 88, 80, 89, 92, 98
|
||||||
],
|
],
|
||||||
"Warszawa": [10, 11, 19, 91, 93, 97, 99],
|
"Warszawa (1)": [10, 11, 19, 91, 93, 97, 99],
|
||||||
"Lublin": [20, 22, 29],
|
"Lublin (2)": [20, 22, 29],
|
||||||
"Kraków": [30, 33, 39],
|
"Kraków (3)": [30, 33, 39],
|
||||||
"Sosnowiec": [40, 44, 49, 94],
|
"Sosnowiec (4)": [40, 44, 49, 94],
|
||||||
"Gdańsk": [50, 55, 59, 90, 95, 96],
|
"Gdańsk (5)": [50, 55, 59, 90, 95, 96],
|
||||||
"Wrocław": [66, 60, 69],
|
"Wrocław (6)": [66, 60, 69],
|
||||||
"Poznań": [77, 70, 79],
|
"Poznań (7)": [77, 70, 79],
|
||||||
"Szczecin": [88, 80],
|
"Szczecin (8)": [88, 80],
|
||||||
"Rezerwa": [89, 92, 98]
|
"Rezerwa (9)": [89, 92, 98]
|
||||||
},
|
},
|
||||||
"categories": {
|
"categoriesRules": {
|
||||||
"EI": "2:00-99:2",
|
"EI": [null, "00", "99"],
|
||||||
"MP/RP": "2:050-169:3",
|
"EC": [null, "000", "049"],
|
||||||
"RO": "2:200-999:3",
|
"EN": [null, "000", "049"],
|
||||||
"PW": "2:6;3:0-899:3",
|
|
||||||
"TM": "2:4;3:0-899:3",
|
"RO": [null, "200", "999"],
|
||||||
"TK": "2:3;3:0-899:3",
|
"RP": [null, "050", "169"],
|
||||||
"LT": "2:5;3:0-899:3"
|
"RM": [null, "200", "999"],
|
||||||
|
"RA": [null, "200", "999"],
|
||||||
|
|
||||||
|
"MO": [null, "200", "999"],
|
||||||
|
"MP": [null, "050", "169"],
|
||||||
|
"MM": [null, "001", "049"],
|
||||||
|
"MH": [null, "170", "199"],
|
||||||
|
|
||||||
|
"PW": ["6", "000", "899"],
|
||||||
|
"PX": ["6", "000", "899"],
|
||||||
|
|
||||||
|
"TC": ["0", "000", "899"],
|
||||||
|
"TG": ["1", "000", "899"],
|
||||||
|
"TR": ["1", "000", "899"],
|
||||||
|
"TD": ["2", "000", "899"],
|
||||||
|
"TK": ["3", "000", "899"],
|
||||||
|
"TN": ["3", "000", "899"],
|
||||||
|
"TM": ["4", "000", "899"],
|
||||||
|
"TS": ["5", "000", "899"],
|
||||||
|
|
||||||
|
"LT": ["5", "000", "899"],
|
||||||
|
"LP": ["6", "000", "899"],
|
||||||
|
"LS": ["9", "000", "899"],
|
||||||
|
|
||||||
|
"ZN": ["9", "000", "899"]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,66 +1,82 @@
|
|||||||
{
|
{
|
||||||
"EU07": {
|
"EU07": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"650": 125
|
"650000": 125
|
||||||
},
|
},
|
||||||
"cargo": {
|
"cargo": {
|
||||||
"2000": 70
|
"2000000": 70
|
||||||
}
|
},
|
||||||
|
"none": 110
|
||||||
|
},
|
||||||
|
"4E": {
|
||||||
|
"passenger": {
|
||||||
|
"650000": 125
|
||||||
|
},
|
||||||
|
"cargo": {
|
||||||
|
"2000000": 70
|
||||||
|
},
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"EU07E": {
|
"EU07E": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"650": 125
|
"650000": 125
|
||||||
},
|
},
|
||||||
"cargo": {
|
"cargo": {
|
||||||
"2000": 70
|
"2000000": 70
|
||||||
}
|
},
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"EP07": {
|
"EP07": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"650": 125
|
"650000": 125
|
||||||
},
|
},
|
||||||
"cargo": null
|
"cargo": null,
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"EP08": {
|
"EP08": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"650": 140
|
"650000": 140
|
||||||
},
|
},
|
||||||
"cargo": null
|
"cargo": null,
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"EP09": {
|
"EP09": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"650": 160
|
"650000": 160
|
||||||
},
|
},
|
||||||
"cargo": null
|
"cargo": null,
|
||||||
|
"none": 160
|
||||||
},
|
},
|
||||||
"ET41": {
|
"ET41": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"700": 125
|
"700000": 125
|
||||||
},
|
},
|
||||||
"cargo": {
|
"cargo": {
|
||||||
"4000": 70
|
"4000000": 70
|
||||||
}
|
},
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"SM42": {
|
"SM42": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"95": 90,
|
"95000": 90,
|
||||||
"200": 80,
|
"200000": 80,
|
||||||
"300": 70,
|
"300000": 70,
|
||||||
"450": 60,
|
"450000": 60,
|
||||||
"750": 50,
|
"750000": 50,
|
||||||
"1130": 40,
|
"1130000": 40,
|
||||||
"1720": 30,
|
"1720000": 30,
|
||||||
"2400": 20
|
"2400000": 20
|
||||||
},
|
},
|
||||||
"cargo": {
|
"cargo": {
|
||||||
"95": 90,
|
"95000": 90,
|
||||||
"200": 80,
|
"200000": 80,
|
||||||
"300": 70,
|
"300000": 70,
|
||||||
"450": 60,
|
"450000": 60,
|
||||||
"750": 50,
|
"750000": 50,
|
||||||
"1130": 40,
|
"1130000": 40,
|
||||||
"1720": 30,
|
"1720000": 30,
|
||||||
"2400": 20
|
"2400000": 20
|
||||||
}
|
},
|
||||||
|
"none": 90
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +0,0 @@
|
|||||||
export const enum EVehicleUseType {
|
|
||||||
LOCO_ELECTRICAL = "loco-e",
|
|
||||||
LOCO_DIESEL = "loco-s",
|
|
||||||
EMU = "loco-ezt",
|
|
||||||
DMU = "loco-szt",
|
|
||||||
|
|
||||||
CAR_PASSENGER = "car-passenger",
|
|
||||||
CAR_CARGO = "car-cargo",
|
|
||||||
}
|
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
import axios from "axios";
|
import axios from 'axios';
|
||||||
|
|
||||||
const http = axios.create({
|
const http = axios.create({
|
||||||
baseURL:
|
baseURL:
|
||||||
import.meta.env.VITE_API_DEV === "1" && import.meta.env.DEV
|
import.meta.env.VITE_API_DEV === '1' && import.meta.env.DEV
|
||||||
? "http://localhost:5500"
|
? 'http://localhost:3001'
|
||||||
: "https://spythere.github.io/api",
|
: 'https://stacjownik.spythere.eu',
|
||||||
});
|
});
|
||||||
|
|
||||||
export default http;
|
export default http;
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
import localePL from "./locales/pl.json";
|
import localePL from './locales/pl.json';
|
||||||
import localeEN from "./locales/en.json";
|
import localeEN from './locales/en.json';
|
||||||
import { createI18n } from "vue-i18n";
|
import { createI18n } from 'vue-i18n';
|
||||||
import http from "./http";
|
|
||||||
|
|
||||||
type LocaleMessageSchema = typeof localePL;
|
type LocaleMessageSchema = typeof localePL;
|
||||||
type LocaleKey = "en" | "pl";
|
type LocaleKey = 'en' | 'pl';
|
||||||
|
|
||||||
const locales: { [key in LocaleKey]: LocaleMessageSchema } = {
|
const locales: { [key in LocaleKey]: LocaleMessageSchema } = {
|
||||||
en: localeEN,
|
en: localeEN,
|
||||||
@@ -12,24 +11,14 @@ const locales: { [key in LocaleKey]: LocaleMessageSchema } = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const locale =
|
const locale =
|
||||||
window.localStorage.getItem("locale") ||
|
window.localStorage.getItem('locale') || (/^pl\b/.test(navigator.language) ? 'pl' : 'en');
|
||||||
(/^pl\b/.test(navigator.language) ? "pl" : "en");
|
|
||||||
|
|
||||||
const i18n = createI18n<[LocaleMessageSchema], "en" | "pl">({
|
const i18n = createI18n<[LocaleMessageSchema], 'en' | 'pl'>({
|
||||||
locale,
|
locale,
|
||||||
fallbackLocale: "pl",
|
fallbackLocale: 'pl',
|
||||||
legacy: false,
|
legacy: false,
|
||||||
globalInjection: true,
|
globalInjection: true,
|
||||||
messages: locales,
|
messages: locales,
|
||||||
});
|
});
|
||||||
|
|
||||||
async function fetchBackendTranslations() {
|
|
||||||
const localeData = (await http.get(`td2/data/locales.json`)).data;
|
|
||||||
|
|
||||||
i18n.global.mergeLocaleMessage("pl", localeData.pl);
|
|
||||||
i18n.global.mergeLocaleMessage("en", localeData.en);
|
|
||||||
}
|
|
||||||
|
|
||||||
fetchBackendTranslations();
|
|
||||||
|
|
||||||
export default i18n;
|
export default i18n;
|
||||||
|
|||||||
@@ -15,12 +15,12 @@
|
|||||||
"cargo-title": "Cargo (only selected freight cars)",
|
"cargo-title": "Cargo (only selected freight cars)",
|
||||||
"no-cargo-available": "no cargo available",
|
"no-cargo-available": "no cargo available",
|
||||||
"cargo-empty": "empty",
|
"cargo-empty": "empty",
|
||||||
"loco-e": "ELECTR.",
|
"loco-electric": "ELECTR.",
|
||||||
"loco-s": "DIESEL",
|
"loco-diesel": "DIESEL",
|
||||||
"loco-ezt": "EMU",
|
"unit-electric": "EMU",
|
||||||
"loco-szt": "DMU",
|
"unit-diesel": "DMU",
|
||||||
"car-passenger": "PASSENGER",
|
"wagon-passenger": "PASSENGER",
|
||||||
"car-cargo": "FREIGHT",
|
"wagon-freight": "FREIGHT",
|
||||||
"action-add": "ADD NEW",
|
"action-add": "ADD NEW",
|
||||||
"action-swap": "SWAP WITH",
|
"action-swap": "SWAP WITH",
|
||||||
"real-stock": "POLISH TRAIN COMPOSITIONS"
|
"real-stock": "POLISH TRAIN COMPOSITIONS"
|
||||||
@@ -30,12 +30,13 @@
|
|||||||
"loading": "IMAGE LOADING...",
|
"loading": "IMAGE LOADING...",
|
||||||
"desc": "Choose a railway vehicle above to see its preview",
|
"desc": "Choose a railway vehicle above to see its preview",
|
||||||
"sponsor-only": "* SPONSORS ONLY UNTIL {0}",
|
"sponsor-only": "* SPONSORS ONLY UNTIL {0}",
|
||||||
"loco-e": "ELECTRIC LOCO",
|
"team-only": "* TD2 TEAM ONLY",
|
||||||
"loco-s": "DIESEL LOCO",
|
"loco-electric": "ELECTRIC LOCO",
|
||||||
"loco-ezt": "ELECTRIC M.U.",
|
"loco-diesel": "DIESEL LOCO",
|
||||||
"loco-szt": "DIESEL M.U.",
|
"unit-electric": "ELECTRIC M.U.",
|
||||||
"car-passenger": "PASSENGER CARRIAGE",
|
"unit-diesel": "DIESEL M.U.",
|
||||||
"car-cargo": "FREIGHT CARRIAGE",
|
"wagon-passenger": "PASSENGER CARRIAGE",
|
||||||
|
"wagon-freight": "FREIGHT CARRIAGE",
|
||||||
"cabin": "Cabin type:",
|
"cabin": "Cabin type:",
|
||||||
"construction": "Construction type:"
|
"construction": "Construction type:"
|
||||||
},
|
},
|
||||||
@@ -47,6 +48,7 @@
|
|||||||
},
|
},
|
||||||
"stocklist": {
|
"stocklist": {
|
||||||
"title": "STOCK EDITOR",
|
"title": "STOCK EDITOR",
|
||||||
|
"disclaimer": "Theorethical value based on vehicles maximum speed in the current composition. It may be inaccurate in relation to the correct operational speed in certain configurations.",
|
||||||
"alert-copied": "The rolling stock has been copied to your clipboard!",
|
"alert-copied": "The rolling stock has been copied to your clipboard!",
|
||||||
"alert-empty": "Lista pojazdów jest pusta!",
|
"alert-empty": "Lista pojazdów jest pusta!",
|
||||||
"prompt-file": "Name a file and download it to the Presets folder (Documents/TTSK/TrainDriver2):",
|
"prompt-file": "Name a file and download it to the Presets folder (Documents/TTSK/TrainDriver2):",
|
||||||
@@ -67,11 +69,13 @@
|
|||||||
"coldstart-info": "Locomotive cold start",
|
"coldstart-info": "Locomotive cold start",
|
||||||
"doublemanning-info": "Double manning",
|
"doublemanning-info": "Double manning",
|
||||||
"list-empty": "Stock list is empty!",
|
"list-empty": "Stock list is empty!",
|
||||||
"warning-not-suitable": "EP07 & EP08 type locomotives are designed for passenger traffic only!",
|
|
||||||
|
"warning-not-suitable": "EP series locomotives are designed for passenger traffic only!",
|
||||||
"warning-passenger-too-long": "Maximum length of a passenger train may not be greater than 350m!",
|
"warning-passenger-too-long": "Maximum length of a passenger train may not be greater than 350m!",
|
||||||
"warning-freight-too-long": "Maximum length of a freight train may not be greater than 650m!",
|
"warning-freight-too-long": "Maximum length of a freight train may not be greater than 650m!",
|
||||||
"warning-too-many-locos": "This train has too many traction units!",
|
"warning-too-many-locos": "This train has too many traction units!",
|
||||||
"warning-too-heavy": "This train is too heavy! Check {href}",
|
"warning-too-heavy": "This train is too heavy! Check {href}",
|
||||||
|
"warning-team-only-vehicle": "There's at least one vehicle available only for TD2 team members in your stock composition! ({0})",
|
||||||
"acceptable-mass-docs": "acceptable rolling stock masses (PL)"
|
"acceptable-mass-docs": "acceptable rolling stock masses (PL)"
|
||||||
},
|
},
|
||||||
"stockgen": {
|
"stockgen": {
|
||||||
@@ -92,57 +96,103 @@
|
|||||||
},
|
},
|
||||||
"numgen": {
|
"numgen": {
|
||||||
"title": "TRAIN NUMBER GENERATOR",
|
"title": "TRAIN NUMBER GENERATOR",
|
||||||
|
"subtitle": "Generates real train number based on Polish railway instruction Ir-11",
|
||||||
"alert": "The number has been copied to your clipboard!",
|
"alert": "The number has been copied to your clipboard!",
|
||||||
"start-region": "Beginning construction region",
|
"start-region": "Beginning construction region",
|
||||||
"end-region": "Terminating construction region",
|
"end-region": "Terminating construction region",
|
||||||
"train-category": "Train category",
|
"train-category": "Train category",
|
||||||
"number-info": "Generated train number:",
|
"number-info": "Generated train number:",
|
||||||
"warning": "Choose category and (optionally) construction regions",
|
"warning": "Choose category and (optionally) construction regions",
|
||||||
"td2-wiki": "> Polish rules of train numbering (TD2 wiki)",
|
"td2-wiki": "> Polish rules of train numbering (forum thread)",
|
||||||
"td2-wiki-link": "https://wiki.td2.info.pl/index.php?title=Zasady_numeracji_poci%C4%85g%C3%B3w/en",
|
"td2-wiki-link": "https://td2.info.pl/english-boards/new-train-categories-in-the-simulator/",
|
||||||
"action-random-region": "DRAW REGIONS",
|
"action-random-region": "DRAW REGIONS",
|
||||||
"action-random-number": "DRAW A NUMBER",
|
"action-random-number": "DRAW LAST DIGITS",
|
||||||
"categories": {
|
"action-random-category": "DRAW A CATEGORY",
|
||||||
"EI": "domestic express (EI)",
|
|
||||||
"MP/RP": "(inter)voivodeship bullet (MP/RP)",
|
|
||||||
"RO": "regional passenger (RO)",
|
|
||||||
"PW": "empty passenger (PW)",
|
|
||||||
"TM": "mass transport freight (TM)",
|
|
||||||
"TK": "non-mass transport freight (TK)",
|
|
||||||
"LT": "locomotive alone (LT)"
|
|
||||||
},
|
|
||||||
"rules": {
|
"rules": {
|
||||||
"EI": "4 digits - ends within the range of 00-99",
|
"first-digit": "First digit:",
|
||||||
"MP/RP": "5 digits - ends within the range of 050-169",
|
"second-digit": "Second digit:",
|
||||||
"RO": "5 digits - ends within the range of 200-999",
|
"third-digit": "Third digit:",
|
||||||
"PW": "6 digits - '6' on the 3rd place; ends within the range of 000-899",
|
"two-first-digits": "Two first digits:",
|
||||||
"TM": "6 digits - '4' on the 3rd place; ends within the range of 000-899",
|
"two-last-digits": "Two last digits:",
|
||||||
"TK": "6 digits - '3' on the 3rd place; ends within the range of 000-899",
|
"three-last-digits": "Three last digits:",
|
||||||
"LT": "6 digits - '5' on the 3rd place; ends within the range of 000-899"
|
"from-pool": "from pool of",
|
||||||
|
"for-category": "for category",
|
||||||
|
"for-region": "for region",
|
||||||
|
"for-region-begin": "for the beginning construction region",
|
||||||
|
"for-region-end": "for the terminating construction region",
|
||||||
|
"from-range": "from range of"
|
||||||
|
},
|
||||||
|
|
||||||
|
"categories": {
|
||||||
|
"EI": "EI - domestic express",
|
||||||
|
"EC": "EC - international express",
|
||||||
|
"EN": "EN - domestic night express",
|
||||||
|
|
||||||
|
"MP": "MP - intervoivodeship bullet",
|
||||||
|
"RP": "RP - voivodeship bullet",
|
||||||
|
"MO": "MO - intervoivodeship regio",
|
||||||
|
"RO": "RO - voivodeship regio",
|
||||||
|
|
||||||
|
"MM": "MM - international bullet",
|
||||||
|
"MH": "MH - intervoivodeship bullet (night / hotel)",
|
||||||
|
"RM": "RM - international voivodeship regio",
|
||||||
|
"RA": "RA - voivodeship regio (urban)",
|
||||||
|
|
||||||
|
"PW": "PW - empty passenger",
|
||||||
|
"PX": "PX - empty passenger test drive",
|
||||||
|
|
||||||
|
"TC": "TC - international freight (intermodal)",
|
||||||
|
"TG": "TG - international freight (cargo)",
|
||||||
|
"TR": "TR - international freight (no cargo)",
|
||||||
|
"TD": "TD - domestic freight (intermodal)",
|
||||||
|
"TM": "TM - domestic freight (cargo)",
|
||||||
|
"TN": "TN - domestic freight (no cargo)",
|
||||||
|
"TK": "TK - freight (stations & sidings)",
|
||||||
|
"TS": "TS - empty freight test drive",
|
||||||
|
|
||||||
|
"LT": "LT - locomotive only",
|
||||||
|
"LT-new": "LT - freight locomotive only",
|
||||||
|
|
||||||
|
"LP": "LP - passenger locomotive only",
|
||||||
|
"LS": "LS - shunting locomotive",
|
||||||
|
|
||||||
|
"ZN": "ZN - inspection / diagnostic"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"wiki": {
|
"wiki": {
|
||||||
"title": "LIST OF AVAILABLE VEHICLES",
|
"title": "LIST OF AVAILABLE VEHICLES",
|
||||||
"action-vehicles": "TRACTION UNITS",
|
"labels": {
|
||||||
"action-carriages": "CARRIAGES",
|
"vehicles": "Vehicles",
|
||||||
"search": "Search for a vehicle...",
|
"sort-by": "Sort by",
|
||||||
"header": {
|
"sort-direction": "Sort direction",
|
||||||
"image": "Image",
|
"search-vehicle": "Find a vehicle",
|
||||||
"type": "Name",
|
"search-vehicle-placeholder": "Input a vehicle name"
|
||||||
"group": "Type group",
|
|
||||||
"constructionType": "Construction",
|
|
||||||
"coldStart": "Cold start",
|
|
||||||
"length": "Length",
|
|
||||||
"mass": "Mass",
|
|
||||||
"maxSpeed": "Speed",
|
|
||||||
"cargoCount": "Cargo count"
|
|
||||||
},
|
},
|
||||||
"loco-ezt": "EMU",
|
"filters": {
|
||||||
"loco-szt": "DMU",
|
"vehicles-all": "all",
|
||||||
"loco-s": "Diesel locomotive",
|
"vehicles-traction": "traction units",
|
||||||
"loco-e": "Electric locomotive",
|
"vehicles-wagon": "wagons"
|
||||||
"car-passenger": "Passenger carriage",
|
},
|
||||||
"car-cargo": "Frieght carriage"
|
"sort-by": {
|
||||||
|
"type": "name",
|
||||||
|
"group": "type group",
|
||||||
|
"length": "length",
|
||||||
|
"weight": "mass",
|
||||||
|
"maxSpeed": "speed",
|
||||||
|
"coldStart": "cold start",
|
||||||
|
"cargoCount": "cargo count"
|
||||||
|
},
|
||||||
|
"sort-direction": {
|
||||||
|
"asc": "ascending",
|
||||||
|
"desc": "descending"
|
||||||
|
},
|
||||||
|
"unit-electric": "EMU",
|
||||||
|
"unit-diesel": "DMU",
|
||||||
|
"loco-diesel": "Diesel locomotive",
|
||||||
|
"loco-electric": "Electric locomotive",
|
||||||
|
"wagon-passenger": "Passenger carriage",
|
||||||
|
"wagon-freight": "Frieght carriage"
|
||||||
},
|
},
|
||||||
"realstock": {
|
"realstock": {
|
||||||
"title": "POLISH TRAIN COMPOSITIONS by",
|
"title": "POLISH TRAIN COMPOSITIONS by",
|
||||||
|
|||||||
@@ -9,18 +9,18 @@
|
|||||||
"version-check": "Strona jest kompletna dla wersji {version} symulatora TD2"
|
"version-check": "Strona jest kompletna dla wersji {version} symulatora TD2"
|
||||||
},
|
},
|
||||||
"inputs": {
|
"inputs": {
|
||||||
"title": "WYBIERZ POJAZD SZYNOWY",
|
"title": "WYBIERZ POJAZD",
|
||||||
"input-vehicle": "Wybierz pojazd trakcyjny",
|
"input-vehicle": "Wybierz pojazd trakcyjny",
|
||||||
"input-carwagon": "Wybierz wagon",
|
"input-carwagon": "Wybierz wagon",
|
||||||
"cargo-title": "Ładunek (tylko wybrane towarowe)",
|
"cargo-title": "Ładunek (tylko wybrane towarowe)",
|
||||||
"no-cargo-available": "brak dostępnych ładunków",
|
"no-cargo-available": "brak dostępnych ładunków",
|
||||||
"cargo-empty": "próżny",
|
"cargo-empty": "próżny",
|
||||||
"loco-e": "ELEKTR.",
|
"loco-electric": "ELEKTR.",
|
||||||
"loco-s": "SPAL.",
|
"loco-diesel": "SPAL.",
|
||||||
"loco-ezt": "EZT",
|
"unit-electric": "EZT",
|
||||||
"loco-szt": "SZT",
|
"unit-diesel": "SZT",
|
||||||
"car-passenger": "PASAŻERSKIE",
|
"wagon-passenger": "PASAŻERSKIE",
|
||||||
"car-cargo": "TOWAROWE",
|
"wagon-freight": "TOWAROWE",
|
||||||
"action-add": "DODAJ NOWY",
|
"action-add": "DODAJ NOWY",
|
||||||
"action-swap": "ZAMIEŃ ZA",
|
"action-swap": "ZAMIEŃ ZA",
|
||||||
"real-stock": "REALNE ZESTAWIENIA"
|
"real-stock": "REALNE ZESTAWIENIA"
|
||||||
@@ -30,12 +30,13 @@
|
|||||||
"loading": "ŁADOWANIE OBRAZU...",
|
"loading": "ŁADOWANIE OBRAZU...",
|
||||||
"desc": "Wybierz pojazd lub wagon, aby zobaczyć jego podgląd powyżej",
|
"desc": "Wybierz pojazd lub wagon, aby zobaczyć jego podgląd powyżej",
|
||||||
"sponsor-only": "* TYLKO DLA SPONSORÓW DO {0}",
|
"sponsor-only": "* TYLKO DLA SPONSORÓW DO {0}",
|
||||||
"loco-e": "ELEKTROWÓZ",
|
"team-only": "* TYLKO DLA ZESPOŁU TD2",
|
||||||
"loco-s": "SPALINOWÓZ",
|
"loco-electric": "ELEKTROWÓZ",
|
||||||
"loco-ezt": "EZT",
|
"loco-diesel": "SPALINOWÓZ",
|
||||||
"loco-szt": "SZT",
|
"unit-electric": "EZT",
|
||||||
"car-passenger": "WAGON PASAŻERSKI",
|
"unit-diesel": "SZT",
|
||||||
"car-cargo": "WAGON TOWAROWY",
|
"wagon-passenger": "WAGON PASAŻERSKI",
|
||||||
|
"wagon-freight": "WAGON TOWAROWY",
|
||||||
"cabin": "Typ kabiny:",
|
"cabin": "Typ kabiny:",
|
||||||
"construction": "Typ konstrukcji:"
|
"construction": "Typ konstrukcji:"
|
||||||
},
|
},
|
||||||
@@ -47,6 +48,7 @@
|
|||||||
},
|
},
|
||||||
"stocklist": {
|
"stocklist": {
|
||||||
"title": "EDYTOR SKŁADU",
|
"title": "EDYTOR SKŁADU",
|
||||||
|
"disclaimer": "Wartość poglądowa wzorowana na prędkościach maksymalnych poszczególnych pojazdów w zestawieniu. Może nie zgadzać się z prawdziwymi prędkościami eksploatacyjnymi w konkretnych konfiguracjach.",
|
||||||
"alert-copied": "Skład został skopiowany do twojego schowka!",
|
"alert-copied": "Skład został skopiowany do twojego schowka!",
|
||||||
"alert-empty": "Lista pojazdów jest pusta!",
|
"alert-empty": "Lista pojazdów jest pusta!",
|
||||||
"prompt-file": "Nazwij plik, a następnie pobierz do folderu Presets (Dokumenty/TTSK/TrainDriver2):",
|
"prompt-file": "Nazwij plik, a następnie pobierz do folderu Presets (Dokumenty/TTSK/TrainDriver2):",
|
||||||
@@ -67,11 +69,13 @@
|
|||||||
"coldstart-info": "Zimny start",
|
"coldstart-info": "Zimny start",
|
||||||
"doublemanning-info": "Podwójna obsada",
|
"doublemanning-info": "Podwójna obsada",
|
||||||
"list-empty": "Lista pojazdów jest pusta!",
|
"list-empty": "Lista pojazdów jest pusta!",
|
||||||
"warning-not-suitable": "Lokomotywy EP07 i EP08 są przeznaczone jedynie do ruchu pasażerskiego!",
|
|
||||||
|
"warning-not-suitable": "Lokomotywy serii EP są przeznaczone jedynie do ruchu pasażerskiego!",
|
||||||
"warning-passenger-too-long": "Maksymalna długość składów pasażerskich nie może przekraczać 350m!",
|
"warning-passenger-too-long": "Maksymalna długość składów pasażerskich nie może przekraczać 350m!",
|
||||||
"warning-freight-too-long": "Maksymalna długość składów innych niż pasażerskie nie może przekraczać 650m!",
|
"warning-freight-too-long": "Maksymalna długość składów innych niż pasażerskie nie może przekraczać 650m!",
|
||||||
"warning-too-many-locos": "Ten skład posiada za dużo pojazdów trakcyjnych!",
|
"warning-too-many-locos": "Ten skład posiada za dużo pojazdów trakcyjnych!",
|
||||||
"warning-too-heavy": "Ten skład jest za ciężki! Sprawdź {href}",
|
"warning-too-heavy": "Ten skład jest za ciężki! Sprawdź {href}",
|
||||||
|
"warning-team-only-vehicle": "W zestawieniu znajduje się co najmniej jeden pojazd dostępny tylko dla członków zespołu TD2! ({0})",
|
||||||
"acceptable-mass-docs": "dopuszczalne masy składów"
|
"acceptable-mass-docs": "dopuszczalne masy składów"
|
||||||
},
|
},
|
||||||
"stockgen": {
|
"stockgen": {
|
||||||
@@ -92,57 +96,103 @@
|
|||||||
},
|
},
|
||||||
"numgen": {
|
"numgen": {
|
||||||
"title": "GENERATOR NUMERU POCIĄGU",
|
"title": "GENERATOR NUMERU POCIĄGU",
|
||||||
|
"subtitle": "Generuje realny numer pociągu na podstawie instrukcji Ir-11",
|
||||||
"alert": "Numer został skopiowany do twojego schowka!",
|
"alert": "Numer został skopiowany do twojego schowka!",
|
||||||
"start-region": "Początkowy obszar konstrukcyjny",
|
"start-region": "Obszar początkowy",
|
||||||
"end-region": "Końcowy obszar konstrukcyjny",
|
"end-region": "Obszar końcowy",
|
||||||
"train-category": "Kategoria pociągu",
|
"train-category": "Kategoria pociągu",
|
||||||
"number-info": "Wygenerowany numer pociągu:",
|
"number-info": "Wygenerowany numer pociągu:",
|
||||||
"warning": "Wybierz kategorię oraz (opcjonalnie) obszary konstrukcyjne",
|
"warning": "Wybierz kategorię oraz (opcjonalnie) obszary konstrukcyjne",
|
||||||
"td2-wiki": "> Szczegółowe zasady numeracji (wikipedia TD2)",
|
"td2-wiki": "> Szczegółowe zasady numeracji (wątek forum)",
|
||||||
"td2-wiki-link": "https://wiki.td2.info.pl/index.php?title=Zasady_numeracji_poci%C4%85g%C3%B3w",
|
"td2-wiki-link": "https://td2.info.pl/ogloszenia/nowe-kategorie-pociagow-w-symulatorze/",
|
||||||
"action-random-region": "LOSUJ OBSZARY",
|
"action-random-region": "LOSUJ OBSZARY",
|
||||||
"action-random-number": "LOSUJ NUMER",
|
"action-random-number": "LOSUJ KOŃCÓWKĘ",
|
||||||
"categories": {
|
"action-random-category": "LOSUJ KATEGORIĘ",
|
||||||
"EI": "ekspres krajowy (EI)",
|
|
||||||
"MP/RP": "(między)wojewódzki pośpieszny (MP/RP)",
|
|
||||||
"RO": "wojewódzki osobowy (RO)",
|
|
||||||
"PW": "próżny \"służbowy\" (PW)",
|
|
||||||
"TM": "towarowy do przewozów masowych (TM)",
|
|
||||||
"TK": "towarowy do obsługi stacji (TK)",
|
|
||||||
"LT": "lokomotywa luzem (LT)"
|
|
||||||
},
|
|
||||||
"rules": {
|
"rules": {
|
||||||
"EI": "4 cyfry - końcówka z przedziału 00-99",
|
"first-digit": "Pierwsza cyfra:",
|
||||||
"MP/RP": "5 cyfr - końcówka z przedziału 050-169",
|
"second-digit": "Druga cyfra:",
|
||||||
"RO": "5 cyfr - końcówka z przedziału 200-999",
|
"third-digit": "Trzecia cyfra:",
|
||||||
"PW": "6 cyfr - '6' na 3. miejscu; końcówka z przedziału 000-899",
|
"two-first-digits": "Dwie pierwsze cyfry:",
|
||||||
"TM": "6 cyfr - '4' na 3. miejscu; końcówka z przedziału 000-899",
|
"two-last-digits": "Dwie ostatnie cyfry:",
|
||||||
"TK": "6 cyfr - '3' na 3. miejscu; końcówka z przedziału 000-899",
|
"three-last-digits": "Trzy ostatnie cyfry:",
|
||||||
"LT": "6 cyfr - '5' na 3. miejscu; końcówka z przedziału 000-899"
|
"from-pool": "z puli",
|
||||||
|
"for-category": "dla kategorii",
|
||||||
|
"for-region": "dla obszaru",
|
||||||
|
"for-region-begin": "dla początkowego obszaru konstrukcyjnego",
|
||||||
|
"for-region-end": "dla końcowego obszaru konstrukcyjnego",
|
||||||
|
"from-range": "z przedziału"
|
||||||
|
},
|
||||||
|
|
||||||
|
"categories": {
|
||||||
|
"EI": "EI - ekspres krajowy",
|
||||||
|
"EC": "EC - ekspres międzynarodowy",
|
||||||
|
"EN": "EN - ekspres krajowy nocny",
|
||||||
|
|
||||||
|
"MP": "MP - międzywoj. pośpieszny",
|
||||||
|
"RP": "RP - wojewódzki pośpieszny",
|
||||||
|
"MO": "MO - międzywoj. osobowy",
|
||||||
|
"RO": "RO - wojewódzki osobowy",
|
||||||
|
|
||||||
|
"MM": "MM - międzynar. pośpieszny",
|
||||||
|
"MH": "MH - międzywoj. pośpieszny hotelowy",
|
||||||
|
"RM": "RM - woj. osobowy międzynarodowy",
|
||||||
|
"RA": "RA - woj. osobowy algomeracyjny",
|
||||||
|
|
||||||
|
"PW": "PW - pasażerski próżny - służbowy",
|
||||||
|
"PX": "PX - pasażerski próżny próbny",
|
||||||
|
|
||||||
|
"TC": "TC - towarowy międzynarodowy intermodalny",
|
||||||
|
"TG": "TG - towarowy międzynarodowy masowy",
|
||||||
|
"TR": "TR - towarowy międzynarodowy niemasowy",
|
||||||
|
"TD": "TD - towarowy krajowy intermodalny",
|
||||||
|
"TM": "TM - towarowy krajowy masowy",
|
||||||
|
"TN": "TN - towarowy krajowy niemasowy",
|
||||||
|
"TK": "TK - towarowy (stacje i bocznice)",
|
||||||
|
"TS": "TS - towarowy próżny próbny",
|
||||||
|
|
||||||
|
"LT": "LT - lokomotywa luzem",
|
||||||
|
"LT-new": "LT - lokomotywa towarowa luzem",
|
||||||
|
|
||||||
|
"LP": "LP - lokomotywa pasażerska luzem",
|
||||||
|
"LS": "LS - lokomotywa manewrowa",
|
||||||
|
|
||||||
|
"ZN": "ZN - inspekcyjny / diagnostyczny"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"wiki": {
|
"wiki": {
|
||||||
"title": "LISTA DOSTĘPNYCH POJAZDÓW",
|
"title": "LISTA DOSTĘPNYCH POJAZDÓW",
|
||||||
"action-vehicles": "POJ. TRAKCYJNE",
|
"labels": {
|
||||||
"action-carriages": "WAGONY",
|
"vehicles": "Pojazdy",
|
||||||
"search": "Wyszukaj pojazd...",
|
"sort-by": "Sortuj wg",
|
||||||
"header": {
|
"sort-direction": "Kierunek sortowania",
|
||||||
"image": "Zdjęcie",
|
"search-vehicle": "Wyszukaj pojazd",
|
||||||
"type": "Nazwa",
|
"search-vehicle-placeholder": "Wpisz nazwę pojazdu"
|
||||||
"group": "Rodzaj",
|
|
||||||
"constructionType": "Konstrukcja",
|
|
||||||
"coldStart": "Zimny start",
|
|
||||||
"length": "Długość",
|
|
||||||
"mass": "Masa",
|
|
||||||
"maxSpeed": "Prędkość",
|
|
||||||
"cargoCount": "Ładunki"
|
|
||||||
},
|
},
|
||||||
"loco-ezt": "EZT",
|
"filters": {
|
||||||
"loco-szt": "SZT",
|
"vehicles-all": "wszystkie",
|
||||||
"loco-s": "Spalinowóz",
|
"vehicles-traction": "trakcyjne",
|
||||||
"loco-e": "Elektrowóz",
|
"vehicles-wagon": "wagony"
|
||||||
"car-passenger": "Wagon pasażerski",
|
},
|
||||||
"car-cargo": "Wagon towarowy"
|
"sort-by": {
|
||||||
|
"type": "nazwa",
|
||||||
|
"group": "rodzaj",
|
||||||
|
"length": "długość",
|
||||||
|
"weight": "masa",
|
||||||
|
"maxSpeed": "prędkość",
|
||||||
|
"coldStart": "zimny start",
|
||||||
|
"cargoCount": "ładunki"
|
||||||
|
},
|
||||||
|
"sort-direction": {
|
||||||
|
"asc": "rosnąco",
|
||||||
|
"desc": "malejąco"
|
||||||
|
},
|
||||||
|
"loco-diesel": "Spalinowóz",
|
||||||
|
"loco-electric": "Elektrowóz",
|
||||||
|
"unit-electric": "EZT",
|
||||||
|
"unit-diesel": "SZT",
|
||||||
|
"wagon-passenger": "Wagon pasażerski",
|
||||||
|
"wagon-freight": "Wagon towarowy"
|
||||||
},
|
},
|
||||||
"realstock": {
|
"realstock": {
|
||||||
"title": "ZESTAWIENIA REALNE by",
|
"title": "ZESTAWIENIA REALNE by",
|
||||||
|
|||||||
@@ -1,17 +1,13 @@
|
|||||||
import { defineComponent } from "vue";
|
import { defineComponent } from 'vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
methods: {
|
methods: {
|
||||||
getIconURL(name: string, ext = "svg"): string {
|
getIconURL(name: string, ext = 'svg'): string {
|
||||||
return `/images/icon-${name}.${ext}`;
|
return `/images/icon-${name}.${ext}`;
|
||||||
},
|
},
|
||||||
|
|
||||||
getThumbnailURL(vehicleType: string, size: "small" | "large") {
|
getThumbnailURL(vehicleType: string, size: 'small' | 'large') {
|
||||||
return `${
|
return `https://static.spythere.eu/images/${vehicleType}--${size == 'small' ? 300 : 800}px.jpg`;
|
||||||
import.meta.env.VITE_API_DEV === "1"
|
|
||||||
? "http://localhost:5500"
|
|
||||||
: "https://spythere.github.io/api"
|
|
||||||
}/td2/images/${vehicleType}--${size == "small" ? 300 : 800}px.jpg`;
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { defineComponent } from 'vue';
|
import { defineComponent } from 'vue';
|
||||||
import { useStore } from '../store';
|
import { useStore } from '../store';
|
||||||
import { ICargo, ICarWagon, ILocomotive, IStock, Vehicle } from '../types';
|
import { ICarWagon, ILocomotive, IStock, ICargo, IVehicle } from '../types';
|
||||||
import { isLocomotive } from '../utils/vehicleUtils';
|
import { isTractionUnit } from '../utils/vehicleUtils';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
setup() {
|
setup() {
|
||||||
@@ -15,39 +15,33 @@ export default defineComponent({
|
|||||||
return `${Math.random().toString(36).slice(5)}`;
|
return `${Math.random().toString(36).slice(5)}`;
|
||||||
},
|
},
|
||||||
|
|
||||||
getStockObject(vehicle: Vehicle, cargo?: ICargo | null, count = 1): IStock {
|
getStockObject(vehicle: IVehicle, cargo?: ICargo | null): IStock {
|
||||||
const isLoco = isLocomotive(vehicle);
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
id: this.getStockId(),
|
id: this.getStockId(),
|
||||||
type: vehicle.type,
|
vehicleRef: vehicle,
|
||||||
length: vehicle.length,
|
cargo: !isTractionUnit(vehicle) && vehicle.loadable && cargo ? cargo : undefined,
|
||||||
mass: vehicle.mass,
|
|
||||||
maxSpeed: vehicle.maxSpeed,
|
|
||||||
isLoco,
|
|
||||||
cargo: !isLoco && vehicle.loadable && cargo ? cargo : undefined,
|
|
||||||
count,
|
|
||||||
imgSrc: vehicle.imageSrc,
|
|
||||||
useType: isLoco ? vehicle.power : vehicle.useType,
|
|
||||||
isSponsorsOnly: vehicle.isSponsorsOnly,
|
|
||||||
constructionType: vehicle.constructionType,
|
|
||||||
sponsorsOnlyTimestamp: vehicle.sponsorsOnlyTimestamp,
|
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
addVehicle(vehicle: Vehicle | null, cargo?: ICargo | null) {
|
addVehicle(vehicle: IVehicle | null, cargo?: ICargo | null) {
|
||||||
if (!vehicle) return;
|
if (!vehicle) return;
|
||||||
|
|
||||||
const stock = this.getStockObject(vehicle, cargo);
|
const stock = this.getStockObject(vehicle, cargo);
|
||||||
|
|
||||||
if (stock.isLoco && !this.store.stockList[0]?.isLoco) this.store.stockList.unshift(stock);
|
if (
|
||||||
|
isTractionUnit(stock.vehicleRef) &&
|
||||||
|
this.store.stockList.length > 0 &&
|
||||||
|
!isTractionUnit(this.store.stockList[0].vehicleRef)
|
||||||
|
)
|
||||||
|
this.store.stockList.unshift(stock);
|
||||||
else this.store.stockList.push(stock);
|
else this.store.stockList.push(stock);
|
||||||
},
|
},
|
||||||
|
|
||||||
addLocomotive(loco: ILocomotive) {
|
addLocomotive(loco: ILocomotive) {
|
||||||
const stockObj = this.getStockObject(loco);
|
const stockObj = this.getStockObject(loco);
|
||||||
|
|
||||||
if (this.store.stockList.length > 0 && !this.store.stockList[0].isLoco) this.store.stockList.unshift(stockObj);
|
if (this.store.stockList.length > 0 && !isTractionUnit(this.store.stockList[0].vehicleRef))
|
||||||
|
this.store.stockList.unshift(stockObj);
|
||||||
else this.store.stockList.push(stockObj);
|
else this.store.stockList.push(stockObj);
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -70,12 +64,14 @@ export default defineComponent({
|
|||||||
this.store.swapVehicles = false;
|
this.store.swapVehicles = false;
|
||||||
|
|
||||||
stockArray.forEach((type, i) => {
|
stockArray.forEach((type, i) => {
|
||||||
let vehicle: Vehicle | null = null;
|
let vehicle: IVehicle | null = null;
|
||||||
let vehicleCargo: ICargo | null = null;
|
let vehicleCargo: ICargo | null = null;
|
||||||
|
|
||||||
const isLoco = /^(EU|EP|ET|SM|EN|2EN|SN)/.test(type);
|
const isTractionUnit = /^([a-zA-Z\d]{0,}-\d{0,})/.test(type);
|
||||||
|
|
||||||
if (isLoco) {
|
console.log(type, isTractionUnit);
|
||||||
|
|
||||||
|
if (isTractionUnit) {
|
||||||
const [locoType, spawnProps] = type.split(',');
|
const [locoType, spawnProps] = type.split(',');
|
||||||
vehicle = this.store.locoDataList.find((loco) => loco.type == locoType) || null;
|
vehicle = this.store.locoDataList.find((loco) => loco.type == locoType) || null;
|
||||||
|
|
||||||
@@ -88,10 +84,10 @@ export default defineComponent({
|
|||||||
const [carType, cargo] = type.split(':');
|
const [carType, cargo] = type.split(':');
|
||||||
vehicle = this.store.carDataList.find((car) => car.type == carType) || null;
|
vehicle = this.store.carDataList.find((car) => car.type == carType) || null;
|
||||||
|
|
||||||
if (cargo) vehicleCargo = vehicle?.cargoList.find((c) => c.id == cargo) || null;
|
if (cargo) vehicleCargo = vehicle?.cargoTypes.find((c) => c.id == cargo) || null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!vehicle) console.log('Brak pojazdu / rodzaj pojazdu źle wczytany:', type);
|
if (!vehicle) console.warn('Brak pojazdu / rodzaj pojazdu źle wczytany:', type);
|
||||||
|
|
||||||
this.addVehicle(vehicle, vehicleCargo);
|
this.addVehicle(vehicle, vehicleCargo);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { defineComponent } from 'vue';
|
import { defineComponent } from 'vue';
|
||||||
import { useStore } from '../store';
|
import { useStore } from '../store';
|
||||||
import { ICarWagon, ILocomotive, IStock, Vehicle } from '../types';
|
import { ICarWagon, ILocomotive, IStock, IVehicle, LocoGroupType, WagonGroupType } from '../types';
|
||||||
import { isLocomotive } from '../utils/vehicleUtils';
|
import { isTractionUnit } from '../utils/vehicleUtils';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
setup() {
|
setup() {
|
||||||
@@ -14,40 +14,37 @@ export default defineComponent({
|
|||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
previewStock(stock: IStock) {
|
previewStock(stock: IStock) {
|
||||||
if (this.store.chosenVehicle?.imageSrc != stock.imgSrc) this.store.imageLoading = true;
|
const vehicleRef = stock.vehicleRef;
|
||||||
|
|
||||||
if (stock.isLoco) {
|
this.store.chosenVehicle = vehicleRef;
|
||||||
const chosenLoco = this.store.locoDataList.find((v) => v.type == stock.type) || null;
|
|
||||||
this.store.chosenVehicle = chosenLoco;
|
if (isTractionUnit(vehicleRef)) {
|
||||||
this.store.chosenLoco = chosenLoco;
|
this.store.chosenLoco = vehicleRef;
|
||||||
this.store.chosenCargo = null;
|
this.store.chosenCargo = null;
|
||||||
this.store.chosenLocoPower = stock.useType;
|
this.store.chosenLocoGroup = vehicleRef.group as LocoGroupType;
|
||||||
} else {
|
} else {
|
||||||
const chosenCar = this.store.carDataList.find((v) => v.type == stock.type) || null;
|
this.store.chosenCar = vehicleRef;
|
||||||
this.store.chosenVehicle = chosenCar;
|
|
||||||
this.store.chosenCar = chosenCar;
|
|
||||||
|
|
||||||
this.store.chosenCargo = stock.cargo || null;
|
this.store.chosenCargo = stock.cargo || null;
|
||||||
this.store.chosenCarUseType = stock.useType;
|
this.store.chosenCarGroup = vehicleRef.group as WagonGroupType;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
previewLocomotive(loco: ILocomotive) {
|
previewLocomotive(loco: ILocomotive) {
|
||||||
this.store.chosenLoco = loco;
|
this.store.chosenLoco = loco;
|
||||||
this.store.chosenVehicle = loco;
|
this.store.chosenVehicle = loco;
|
||||||
this.store.chosenLocoPower = loco.power;
|
this.store.chosenLocoGroup = loco.group;
|
||||||
},
|
},
|
||||||
|
|
||||||
previewCarWagon(carWagon: ICarWagon) {
|
previewCarWagon(carWagon: ICarWagon) {
|
||||||
this.store.chosenCar = carWagon;
|
this.store.chosenCar = carWagon;
|
||||||
this.store.chosenCarUseType = carWagon.useType;
|
this.store.chosenCarGroup = carWagon.group;
|
||||||
this.store.chosenVehicle = carWagon;
|
this.store.chosenVehicle = carWagon;
|
||||||
|
|
||||||
this.store.chosenCargo = null;
|
this.store.chosenCargo = null;
|
||||||
},
|
},
|
||||||
|
|
||||||
previewVehicle(vehicle: Vehicle) {
|
previewVehicle(vehicle: IVehicle) {
|
||||||
if (isLocomotive(vehicle)) this.previewLocomotive(vehicle);
|
if (isTractionUnit(vehicle)) this.previewLocomotive(vehicle);
|
||||||
else this.previewCarWagon(vehicle);
|
else this.previewCarWagon(vehicle);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|||||||
@@ -1,47 +0,0 @@
|
|||||||
import { defineComponent } from "vue";
|
|
||||||
import { useStore } from "../store";
|
|
||||||
|
|
||||||
export default defineComponent({
|
|
||||||
setup() {
|
|
||||||
const store = useStore();
|
|
||||||
|
|
||||||
return {
|
|
||||||
store,
|
|
||||||
};
|
|
||||||
},
|
|
||||||
computed: {
|
|
||||||
trainTooLong() {
|
|
||||||
return (
|
|
||||||
(this.store.totalLength > 350 && this.store.isTrainPassenger) ||
|
|
||||||
(this.store.totalLength > 650 && !this.store.isTrainPassenger)
|
|
||||||
);
|
|
||||||
},
|
|
||||||
|
|
||||||
trainTooHeavy() {
|
|
||||||
return (
|
|
||||||
this.store.acceptableMass &&
|
|
||||||
this.store.totalMass > this.store.acceptableMass
|
|
||||||
);
|
|
||||||
},
|
|
||||||
|
|
||||||
locoNotSuitable() {
|
|
||||||
return (
|
|
||||||
!this.store.isTrainPassenger &&
|
|
||||||
this.store.stockList.length > 1 &&
|
|
||||||
!this.store.stockList.every((stock) => stock.isLoco) &&
|
|
||||||
this.store.stockList.some(
|
|
||||||
(stock) => stock.isLoco && stock.type.startsWith("EP"),
|
|
||||||
)
|
|
||||||
);
|
|
||||||
},
|
|
||||||
|
|
||||||
tooManyLocomotives() {
|
|
||||||
return (
|
|
||||||
this.store.stockList.reduce((acc, stock) => {
|
|
||||||
if (stock.isLoco) acc += stock.count;
|
|
||||||
return acc;
|
|
||||||
}, 0) > 2
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
@@ -1,91 +1,149 @@
|
|||||||
import { IStockData, IStore } from './types';
|
import {
|
||||||
|
IVehiclesData,
|
||||||
|
ICarWagon,
|
||||||
|
ILocomotive,
|
||||||
|
ICargo,
|
||||||
|
IVehicle,
|
||||||
|
IStock,
|
||||||
|
IRealComposition,
|
||||||
|
LocoGroupType,
|
||||||
|
WagonGroupType,
|
||||||
|
} from './types';
|
||||||
import { defineStore } from 'pinia';
|
import { defineStore } from 'pinia';
|
||||||
import {
|
import {
|
||||||
acceptableMass,
|
acceptableWeight,
|
||||||
carDataList,
|
carDataList,
|
||||||
chosenRealStock,
|
isTractionUnit,
|
||||||
isTrainPassenger,
|
isTrainPassenger,
|
||||||
locoDataList,
|
locoDataList,
|
||||||
maxStockSpeed,
|
maxStockSpeed,
|
||||||
totalLength,
|
totalLength,
|
||||||
totalMass,
|
totalWeight,
|
||||||
} from './utils/vehicleUtils';
|
} from './utils/vehicleUtils';
|
||||||
|
|
||||||
|
import i18n from './i18n-setup';
|
||||||
import http from './http';
|
import http from './http';
|
||||||
|
|
||||||
export const useStore = defineStore({
|
export const useStore = defineStore({
|
||||||
id: 'store',
|
id: 'store',
|
||||||
state: () =>
|
state: () => ({
|
||||||
({
|
chosenCar: null as ICarWagon | null,
|
||||||
chosenCar: null,
|
chosenLoco: null as ILocomotive | null,
|
||||||
chosenLoco: null,
|
chosenCargo: null as ICargo | null,
|
||||||
chosenCargo: null,
|
chosenVehicle: null as IVehicle | null,
|
||||||
chosenVehicle: null,
|
|
||||||
|
|
||||||
isColdStart: false,
|
isColdStart: false,
|
||||||
isDoubleManned: false,
|
isDoubleManned: false,
|
||||||
|
|
||||||
showSupporter: false,
|
chosenLocoGroup: 'loco-electric' as LocoGroupType,
|
||||||
imageLoading: false,
|
chosenCarGroup: 'wagon-passenger' as WagonGroupType,
|
||||||
|
|
||||||
chosenLocoPower: 'loco-e',
|
stockList: [] as IStock[],
|
||||||
chosenCarUseType: 'car-passenger',
|
cargoOptions: [] as any[][],
|
||||||
|
|
||||||
stockList: [],
|
swapVehicles: false,
|
||||||
cargoOptions: [],
|
|
||||||
|
|
||||||
readyStockList: [],
|
chosenStockListIndex: -1,
|
||||||
|
|
||||||
swapVehicles: false,
|
vehiclePreviewSrc: '',
|
||||||
|
|
||||||
chosenStockListIndex: -1,
|
stockSectionMode: 'stock-list',
|
||||||
chosenRealStockName: undefined,
|
|
||||||
|
|
||||||
vehiclePreviewSrc: '',
|
isRandomizerCardOpen: false,
|
||||||
|
isRealStockListCardOpen: false,
|
||||||
|
|
||||||
stockSectionMode: 'stock-list',
|
vehiclesData: undefined as IVehiclesData | undefined,
|
||||||
|
|
||||||
isRandomizerCardOpen: false,
|
lastFocusedElement: null as HTMLElement | null,
|
||||||
isRealStockListCardOpen: false,
|
}),
|
||||||
|
|
||||||
stockData: undefined,
|
|
||||||
|
|
||||||
lastFocusedElement: null,
|
|
||||||
}) as IStore,
|
|
||||||
|
|
||||||
getters: {
|
getters: {
|
||||||
locoDataList: (state) => locoDataList(state),
|
locoDataList: (state) => locoDataList(state.vehiclesData),
|
||||||
carDataList: (state) => carDataList(state),
|
carDataList: (state) => carDataList(state.vehiclesData),
|
||||||
vehicleDataList: (state) => [...locoDataList(state), ...carDataList(state)],
|
vehicleDataList: (state) => [
|
||||||
totalMass: (state) => totalMass(state),
|
...locoDataList(state.vehiclesData),
|
||||||
totalLength: (state) => totalLength(state),
|
...carDataList(state.vehiclesData),
|
||||||
maxStockSpeed: (state) => maxStockSpeed(state),
|
],
|
||||||
isTrainPassenger: (state) => isTrainPassenger(state),
|
totalWeight: (state) => totalWeight(state.stockList),
|
||||||
chosenRealStock: (state) => chosenRealStock(state),
|
totalLength: (state) => totalLength(state.stockList),
|
||||||
acceptableMass: (state) => acceptableMass(state),
|
maxStockSpeed: (state) => maxStockSpeed(state.stockList),
|
||||||
|
isTrainPassenger: (state) => isTrainPassenger(state.stockList),
|
||||||
|
acceptableWeight: (state) => acceptableWeight(state.stockList),
|
||||||
|
|
||||||
|
realCompositionList: (state) => {
|
||||||
|
if (!state.vehiclesData) return [];
|
||||||
|
|
||||||
|
return Object.keys(state.vehiclesData.realCompositions).reduce<IRealComposition[]>(
|
||||||
|
(acc, key) => {
|
||||||
|
const [type, number, ...name] = key.split(' ');
|
||||||
|
|
||||||
|
const obj = {
|
||||||
|
number: number.replace(/_/g, '/'),
|
||||||
|
name: name.join(' '),
|
||||||
|
stockString: state.vehiclesData!.realCompositions[key],
|
||||||
|
type,
|
||||||
|
};
|
||||||
|
|
||||||
|
acc.push({
|
||||||
|
stockId: `${obj.type} ${obj.number} ${obj.name}`,
|
||||||
|
...obj,
|
||||||
|
});
|
||||||
|
|
||||||
|
return acc;
|
||||||
|
},
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
stockSupportsColdStart: (state) => {
|
stockSupportsColdStart: (state) => {
|
||||||
if (state.stockList.length == 0) return false;
|
if (state.stockList.length == 0) return false;
|
||||||
if (!state.stockList[0].isLoco) return false;
|
|
||||||
|
if (!isTractionUnit(state.stockList[0].vehicleRef)) return false;
|
||||||
|
else if (state.stockList.length > 1) return false;
|
||||||
|
|
||||||
const headingLoco = state.stockList[0];
|
const headingLoco = state.stockList[0];
|
||||||
|
|
||||||
return state.stockData?.props.find((stock) => stock.type == headingLoco.constructionType)?.coldStart ?? false;
|
return (
|
||||||
|
state.vehiclesData?.vehicleProps.find(
|
||||||
|
(stock) => stock.type == headingLoco.vehicleRef.constructionType
|
||||||
|
)?.coldStart ?? false
|
||||||
|
);
|
||||||
},
|
},
|
||||||
|
|
||||||
stockSupportsDoubleManning: (state) => {
|
stockSupportsDoubleManning: (state) => {
|
||||||
if (state.stockList.length == 0) return false;
|
if (state.stockList.length == 0) return false;
|
||||||
if (!state.stockList[0].isLoco) return false;
|
if (!isTractionUnit(state.stockList[0].vehicleRef)) return false;
|
||||||
|
|
||||||
const headingLoco = state.stockList[0];
|
const headingLoco = state.stockList[0];
|
||||||
|
|
||||||
return state.stockData?.props.find((stock) => stock.type == headingLoco.constructionType)?.doubleManned ?? false;
|
return (
|
||||||
|
state.vehiclesData?.vehicleProps.find(
|
||||||
|
(stock) => stock.type == headingLoco.vehicleRef.constructionType
|
||||||
|
)?.doubleManned ?? false
|
||||||
|
);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
actions: {
|
actions: {
|
||||||
async fetchStockInfoData() {
|
async fetchVehiclesAPI() {
|
||||||
const stockData = (await http.get<IStockData>('td2/data/stockInfo.json')).data;
|
try {
|
||||||
this.stockData = stockData;
|
const vehiclesData = (await http.get<IVehiclesData>('/vehicles')).data;
|
||||||
|
this.vehiclesData = vehiclesData;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async setupAPIData() {
|
||||||
|
await this.fetchVehiclesAPI();
|
||||||
|
this.mergeBackendTranslations();
|
||||||
|
},
|
||||||
|
|
||||||
|
async mergeBackendTranslations() {
|
||||||
|
if (!this.vehiclesData) return;
|
||||||
|
|
||||||
|
i18n.global.mergeLocaleMessage('pl', this.vehiclesData.vehicleLocales.pl);
|
||||||
|
i18n.global.mergeLocaleMessage('en', this.vehiclesData.vehicleLocales.en);
|
||||||
},
|
},
|
||||||
|
|
||||||
handleRouting() {
|
handleRouting() {
|
||||||
|
|||||||
@@ -3,9 +3,12 @@ $breakpointSm: 550px;
|
|||||||
|
|
||||||
$bgColor: #2b3552;
|
$bgColor: #2b3552;
|
||||||
$textColor: #fff;
|
$textColor: #fff;
|
||||||
$secondaryColor: #222;
|
$secondaryColor: #1b1b1b;
|
||||||
$accentColor: #e4c428;
|
$accentColor: #e4c428;
|
||||||
|
|
||||||
|
$sponsorColor: gold;
|
||||||
|
$teamColor: #ff4848;
|
||||||
|
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: 'Lato';
|
font-family: 'Lato';
|
||||||
src:
|
src:
|
||||||
@@ -114,11 +117,28 @@ button {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[data-tooltip]:hover::after,
|
||||||
|
[data-tooltip]:focus::after {
|
||||||
|
position: absolute;
|
||||||
|
transform: translateX(10px);
|
||||||
|
|
||||||
|
content: attr(data-tooltip);
|
||||||
|
color: white;
|
||||||
|
background: black;
|
||||||
|
padding: 0.5em;
|
||||||
|
max-width: 300px;
|
||||||
|
z-index: 100;
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-tooltip] {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
.btn {
|
.btn {
|
||||||
padding: 0.4em 0.75em;
|
padding: 0.4em 0.75em;
|
||||||
|
|
||||||
outline: none;
|
outline: none;
|
||||||
background-color: #222;
|
background-color: $secondaryColor;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
|
|
||||||
@@ -190,12 +210,11 @@ input[type='number'] {
|
|||||||
outline: none;
|
outline: none;
|
||||||
|
|
||||||
padding: 0.25em 0.35em;
|
padding: 0.25em 0.35em;
|
||||||
|
height: 100%;
|
||||||
|
|
||||||
color: white;
|
color: white;
|
||||||
font-size: 1em;
|
font-size: 1em;
|
||||||
|
|
||||||
width: 18em;
|
|
||||||
|
|
||||||
&:focus-visible {
|
&:focus-visible {
|
||||||
border-color: $accentColor;
|
border-color: $accentColor;
|
||||||
}
|
}
|
||||||
@@ -227,6 +246,13 @@ ul {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
height: 3px;
|
||||||
|
background-color: white;
|
||||||
|
outline: none;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
.g-card {
|
.g-card {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 1em;
|
top: 1em;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
@import "./global.scss";
|
@import './global.scss';
|
||||||
|
|
||||||
.tab {
|
.tab {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@@ -8,12 +8,18 @@
|
|||||||
padding: 0.5em 1em;
|
padding: 0.5em 1em;
|
||||||
|
|
||||||
background-color: $secondaryColor;
|
background-color: $secondaryColor;
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
h2 {
|
h2 {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
color: white;
|
color: white;
|
||||||
font-size: 1.35em;
|
font-size: 1.35em;
|
||||||
text-align: center;
|
}
|
||||||
|
|
||||||
|
h3 {
|
||||||
|
margin: 0.5em 0 0 0;
|
||||||
|
font-size: 1.15em;
|
||||||
|
color: #aaa;
|
||||||
}
|
}
|
||||||
|
|
||||||
button {
|
button {
|
||||||
@@ -58,13 +64,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
hr {
|
|
||||||
height: 3px;
|
|
||||||
background-color: white;
|
|
||||||
outline: none;
|
|
||||||
margin: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media only screen and (max-width: 470px) {
|
@media only screen and (max-width: 470px) {
|
||||||
.tab_attributes {
|
.tab_attributes {
|
||||||
label {
|
label {
|
||||||
|
|||||||
@@ -1,129 +1,87 @@
|
|||||||
export type Vehicle = ILocomotive | ICarWagon;
|
export type IVehicle = ILocomotive | ICarWagon;
|
||||||
export type StockSectionMode = 'STOCK_LIST' | 'STOCK_GENERATOR';
|
export type StockSectionMode = 'STOCK_LIST' | 'STOCK_GENERATOR';
|
||||||
|
|
||||||
export interface IStore {
|
export type LocoGroupType = 'loco-electric' | 'loco-diesel' | 'unit-electric' | 'unit-diesel';
|
||||||
chosenCar: ICarWagon | null;
|
export type WagonGroupType = 'wagon-passenger' | 'wagon-freight';
|
||||||
chosenLoco: ILocomotive | null;
|
export type VehicleGroupType = LocoGroupType | WagonGroupType;
|
||||||
chosenCargo: ICargo | null;
|
export type RestrictionType = 'sponsorOnly' | 'teamOnly';
|
||||||
chosenVehicle: Vehicle | null;
|
|
||||||
|
|
||||||
isColdStart: boolean;
|
export interface IVehicleProps {
|
||||||
isDoubleManned: boolean;
|
|
||||||
|
|
||||||
showSupporter: boolean;
|
|
||||||
imageLoading: boolean;
|
|
||||||
|
|
||||||
chosenLocoPower: string;
|
|
||||||
chosenCarUseType: string;
|
|
||||||
|
|
||||||
stockList: IStock[];
|
|
||||||
readyStockList: IReadyStockItem[];
|
|
||||||
cargoOptions: any[][];
|
|
||||||
|
|
||||||
chosenStockListIndex: number;
|
|
||||||
chosenRealStockName?: string;
|
|
||||||
|
|
||||||
swapVehicles: boolean;
|
|
||||||
vehiclePreviewSrc: string;
|
|
||||||
|
|
||||||
isRandomizerCardOpen: boolean;
|
|
||||||
isRealStockListCardOpen: boolean;
|
|
||||||
|
|
||||||
stockSectionMode: 'stock-list' | 'stock-generator' | 'number-generator' | 'wiki-list';
|
|
||||||
stockData?: IStockData;
|
|
||||||
|
|
||||||
lastFocusedElement: HTMLElement | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export type TLocoGroup = 'loco-e' | 'loco-s' | 'loco-ezt' | 'loco-szt';
|
|
||||||
export type TCarWagonGroup = 'car-passenger' | 'car-cargo';
|
|
||||||
|
|
||||||
export interface IStockProps {
|
|
||||||
type: string;
|
type: string;
|
||||||
|
speed: number;
|
||||||
length: number;
|
length: number;
|
||||||
mass: number;
|
weight: number;
|
||||||
cargo?: string | null;
|
cargoTypes?: ICargo[];
|
||||||
coldStart?: boolean;
|
coldStart?: boolean;
|
||||||
doubleManned?: boolean;
|
doubleManned?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IStockData {
|
export interface ICargo {
|
||||||
version: string;
|
id: string;
|
||||||
|
weight: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface IVehiclesData {
|
||||||
|
simulatorVersion: string;
|
||||||
|
|
||||||
generator: {
|
generator: {
|
||||||
passenger: any;
|
|
||||||
cargo: {
|
cargo: {
|
||||||
[key: string]: string[];
|
[key: string]: string[];
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
info: {
|
vehicleList: any[][];
|
||||||
'car-cargo': [string, string, boolean, number | null, string][];
|
|
||||||
'car-passenger': [string, string, boolean, number | null, string][];
|
vehicleProps: IVehicleProps[];
|
||||||
'loco-e': [string, string, string, string, number | null][];
|
|
||||||
'loco-s': [string, string, string, string, number | null][];
|
vehicleLocales: {
|
||||||
'loco-szt': [string, string, string, string, number | null][];
|
pl: {
|
||||||
'loco-ezt': [string, string, string, string, number | null][];
|
cargo: Record<string, string>;
|
||||||
|
usage: Record<string, string>;
|
||||||
|
};
|
||||||
|
en: {
|
||||||
|
cargo: Record<string, string>;
|
||||||
|
usage: Record<string, string>;
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
props: IStockProps[];
|
realCompositions: Record<string, string>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ILocomotive {
|
export interface ILocomotive {
|
||||||
type: string;
|
type: string;
|
||||||
power: TLocoGroup;
|
group: LocoGroupType;
|
||||||
group: TLocoGroup;
|
|
||||||
constructionType: string;
|
constructionType: string;
|
||||||
cabinType: string;
|
cabinType: string;
|
||||||
maxSpeed: number;
|
maxSpeed: number;
|
||||||
isSponsorsOnly: boolean;
|
weight: number;
|
||||||
sponsorsOnlyTimestamp: number;
|
|
||||||
imageSrc: string;
|
|
||||||
|
|
||||||
mass: number;
|
|
||||||
length: number;
|
length: number;
|
||||||
coldStart: boolean;
|
coldStart: boolean;
|
||||||
doubleManned: boolean;
|
doubleManned: boolean;
|
||||||
|
sponsorOnlyTimestamp: number;
|
||||||
|
teamOnly: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ICarWagon {
|
export interface ICarWagon {
|
||||||
type: string;
|
type: string;
|
||||||
useType: TCarWagonGroup;
|
group: WagonGroupType;
|
||||||
group: TCarWagonGroup;
|
|
||||||
constructionType: string;
|
constructionType: string;
|
||||||
loadable: boolean;
|
loadable: boolean;
|
||||||
isSponsorsOnly: boolean;
|
|
||||||
sponsorsOnlyTimestamp: number;
|
|
||||||
maxSpeed: number;
|
maxSpeed: number;
|
||||||
imageSrc: string;
|
weight: number;
|
||||||
|
|
||||||
mass: number;
|
|
||||||
length: number;
|
length: number;
|
||||||
cargoList: { id: string; totalMass: number }[];
|
cargoTypes: ICargo[];
|
||||||
}
|
sponsorOnlyTimestamp: number;
|
||||||
|
teamOnly: boolean;
|
||||||
export interface ICargo {
|
|
||||||
id: string;
|
|
||||||
totalMass: number;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IStock {
|
export interface IStock {
|
||||||
id: string;
|
id: string;
|
||||||
type: string;
|
vehicleRef: IVehicle;
|
||||||
useType: string;
|
cargo?: ICargo;
|
||||||
constructionType: string;
|
|
||||||
length: number;
|
|
||||||
mass: number;
|
|
||||||
maxSpeed: number;
|
|
||||||
cargo?: { id: string; totalMass: number };
|
|
||||||
isLoco: boolean;
|
|
||||||
isSponsorsOnly: boolean;
|
|
||||||
sponsorsOnlyTimestamp: number;
|
|
||||||
count: number;
|
|
||||||
imgSrc?: string;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IReadyStockItem {
|
export interface IRealComposition {
|
||||||
stockId: string;
|
stockId: string;
|
||||||
stockString: string;
|
stockString: string;
|
||||||
type: string;
|
type: string;
|
||||||
|
|||||||
@@ -4,19 +4,30 @@ import massLimits from '../constants/massLimits.json';
|
|||||||
export type SpeedLimitLocoType = keyof typeof speedLimits;
|
export type SpeedLimitLocoType = keyof typeof speedLimits;
|
||||||
export type MassLimitLocoType = keyof typeof massLimits;
|
export type MassLimitLocoType = keyof typeof massLimits;
|
||||||
|
|
||||||
export function calculateSpeedLimit(locoType: SpeedLimitLocoType, stockMass: number, isTrainPassenger: boolean) {
|
export function calculateSpeedLimit(
|
||||||
const speedTable = speedLimits[locoType][isTrainPassenger ? 'passenger' : 'cargo'];
|
locoType: SpeedLimitLocoType,
|
||||||
|
stockTotalWeight: number,
|
||||||
|
stockCount: number,
|
||||||
|
isTrainPassenger: boolean
|
||||||
|
) {
|
||||||
|
if (speedLimits[locoType] === undefined) return 0;
|
||||||
|
|
||||||
|
if (stockCount == 1) return speedLimits[locoType]['none'];
|
||||||
|
|
||||||
|
const stockType = isTrainPassenger ? 'passenger' : 'cargo';
|
||||||
|
const speedTable = speedLimits[locoType][stockType];
|
||||||
|
|
||||||
if (!speedTable) return undefined;
|
if (!speedTable) return undefined;
|
||||||
|
|
||||||
let speedLimit = 0;
|
let speedLimit = 0;
|
||||||
for (const mass in speedTable) if (stockMass > Number(mass)) speedLimit = (speedTable as any)[mass];
|
for (const mass in speedTable)
|
||||||
|
if (stockTotalWeight > Number(mass)) speedLimit = (speedTable as any)[mass];
|
||||||
|
|
||||||
return speedLimit;
|
return speedLimit;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function calculateMassLimit(locoType: MassLimitLocoType, isTrainPassenger: boolean) {
|
export function calculateMassLimit(locoType: MassLimitLocoType, isTrainPassenger: boolean) {
|
||||||
if(massLimits[locoType] === undefined) return 0;
|
if (massLimits[locoType] === undefined) return 0;
|
||||||
|
|
||||||
return massLimits[locoType][isTrainPassenger ? 0 : 1] || 0;
|
return massLimits[locoType][isTrainPassenger ? 0 : 1] || 0;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,145 +1,146 @@
|
|||||||
import { EVehicleUseType } from '../enums/EVehicleUseType';
|
import {
|
||||||
import { ICarWagon, ILocomotive, IStore, TCarWagonGroup, TLocoGroup } from '../types';
|
ICarWagon,
|
||||||
import { MassLimitLocoType, SpeedLimitLocoType, calculateMassLimit, calculateSpeedLimit } from './vehicleLimitsUtils';
|
ILocomotive,
|
||||||
|
IStock,
|
||||||
|
IVehiclesData,
|
||||||
|
LocoGroupType,
|
||||||
|
WagonGroupType,
|
||||||
|
} from '../types';
|
||||||
|
import {
|
||||||
|
MassLimitLocoType,
|
||||||
|
SpeedLimitLocoType,
|
||||||
|
calculateMassLimit,
|
||||||
|
calculateSpeedLimit,
|
||||||
|
} from './vehicleLimitsUtils';
|
||||||
|
|
||||||
export function isLocomotive(vehicle: ILocomotive | ICarWagon): vehicle is ILocomotive {
|
export function isTractionUnit(vehicle: ILocomotive | ICarWagon): vehicle is ILocomotive {
|
||||||
return (vehicle as ILocomotive).power !== undefined;
|
return (vehicle as ILocomotive).cabinType !== undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function locoDataList(state: IStore) {
|
export function locoDataList(vehiclesData: IVehiclesData | undefined) {
|
||||||
if (!state.stockData) return [];
|
if (!vehiclesData) return [];
|
||||||
|
|
||||||
const stockData = state.stockData;
|
return vehiclesData.vehicleList.reduce<ILocomotive[]>((acc, vehicleInfoArray) => {
|
||||||
|
// check if data array has 5 elements (locos & units only)
|
||||||
|
if (vehicleInfoArray.length != 5) return acc;
|
||||||
|
|
||||||
return Object.keys(stockData.info).reduce((acc, vehiclePower) => {
|
const [type, constructionType, cabinType, group, restrictions] = vehicleInfoArray;
|
||||||
if (!vehiclePower.startsWith('loco')) return acc;
|
const locoProps = vehiclesData.vehicleProps.find((prop) => constructionType == prop.type);
|
||||||
|
|
||||||
const locoVehiclesData = stockData.info[vehiclePower as TLocoGroup];
|
if (!locoProps) {
|
||||||
|
console.warn('Brak atrybutów dla pojazdu:', type);
|
||||||
|
return acc;
|
||||||
|
}
|
||||||
|
|
||||||
locoVehiclesData.forEach((loco) => {
|
acc.push({
|
||||||
if (state.showSupporter && !loco[4]) return;
|
group: group as LocoGroupType,
|
||||||
|
|
||||||
const [type, constructionType, cabinType, maxSpeed, sponsorsTimestamp] = loco;
|
type,
|
||||||
const locoProps = stockData.props.find((prop) => constructionType == prop.type);
|
constructionType,
|
||||||
|
cabinType,
|
||||||
|
|
||||||
acc.push({
|
sponsorOnlyTimestamp: restrictions?.sponsorOnly ?? 0,
|
||||||
power: vehiclePower as TLocoGroup,
|
teamOnly: restrictions?.teamOnly ?? false,
|
||||||
group: vehiclePower as TLocoGroup,
|
|
||||||
type,
|
|
||||||
constructionType,
|
|
||||||
cabinType,
|
|
||||||
maxSpeed: Number(maxSpeed),
|
|
||||||
isSponsorsOnly: Number(sponsorsTimestamp) > Date.now(),
|
|
||||||
sponsorsOnlyTimestamp: Number(sponsorsTimestamp),
|
|
||||||
imageSrc: '',
|
|
||||||
|
|
||||||
length: locoProps?.length && type.startsWith('2EN') ? locoProps.length * 2 : locoProps?.length ?? 0,
|
maxSpeed: locoProps.speed,
|
||||||
mass: locoProps?.mass && type.startsWith('2EN') ? 253 : locoProps?.mass ?? 0,
|
length: locoProps.length,
|
||||||
|
weight: locoProps.weight,
|
||||||
|
|
||||||
coldStart: locoProps?.coldStart ?? false,
|
coldStart: locoProps.coldStart ?? false,
|
||||||
doubleManned: locoProps?.doubleManned ?? false,
|
doubleManned: locoProps.doubleManned ?? false,
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return acc;
|
return acc;
|
||||||
}, [] as ILocomotive[]);
|
}, []);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function carDataList(state: IStore) {
|
export function carDataList(vehiclesData: IVehiclesData | undefined) {
|
||||||
if (!state.stockData) return [];
|
if (!vehiclesData) return [];
|
||||||
|
|
||||||
const stockData = state.stockData;
|
return vehiclesData.vehicleList.reduce<ICarWagon[]>((acc, vehicleInfoArray) => {
|
||||||
|
// check if data array has 4 elements (wagons only)
|
||||||
|
if (vehicleInfoArray.length != 4) return acc;
|
||||||
|
|
||||||
return Object.keys(stockData.info).reduce((acc, vehicleUseType) => {
|
const [type, constructionType, group, restrictions] = vehicleInfoArray;
|
||||||
if (!vehicleUseType.startsWith('car')) return acc;
|
const wagonProps = vehiclesData.vehicleProps.find((prop) => constructionType == prop.type);
|
||||||
|
|
||||||
const carVehiclesData = stockData.info[vehicleUseType as TCarWagonGroup];
|
if (!wagonProps) {
|
||||||
|
console.warn('Brak atrybutów dla pojazdu:', type);
|
||||||
|
return acc;
|
||||||
|
}
|
||||||
|
|
||||||
carVehiclesData.forEach((car) => {
|
acc.push({
|
||||||
const [type, constructionType, loadable, sponsorsOnlyTimestamp, maxSpeed] = car;
|
group: group as WagonGroupType,
|
||||||
|
type,
|
||||||
|
constructionType,
|
||||||
|
loadable: wagonProps.cargoTypes ? wagonProps.cargoTypes.length > 0 : false,
|
||||||
|
cargoTypes: wagonProps?.cargoTypes ?? [],
|
||||||
|
|
||||||
if (state.showSupporter && Number(sponsorsOnlyTimestamp) <= Date.now()) return;
|
sponsorOnlyTimestamp: restrictions?.sponsorOnly ?? 0,
|
||||||
|
teamOnly: restrictions?.teamOnly ?? false,
|
||||||
|
|
||||||
const carPropsData = stockData.props.find((v) => type.toString().startsWith(v.type));
|
maxSpeed: wagonProps.speed,
|
||||||
|
weight: wagonProps?.weight || 0,
|
||||||
acc.push({
|
length: wagonProps?.length || 0,
|
||||||
useType: vehicleUseType as TCarWagonGroup,
|
|
||||||
group: vehicleUseType as TCarWagonGroup,
|
|
||||||
type,
|
|
||||||
constructionType,
|
|
||||||
loadable,
|
|
||||||
isSponsorsOnly: Number(sponsorsOnlyTimestamp) > Date.now(),
|
|
||||||
sponsorsOnlyTimestamp: Number(sponsorsOnlyTimestamp),
|
|
||||||
maxSpeed: Number(maxSpeed),
|
|
||||||
imageSrc: '',
|
|
||||||
cargoList:
|
|
||||||
carPropsData?.cargo?.split(';').map((cargo) => ({
|
|
||||||
id: cargo.split(':')[0],
|
|
||||||
totalMass: Number(cargo.split(':')[1]),
|
|
||||||
})) || [],
|
|
||||||
|
|
||||||
mass: carPropsData?.mass || 0,
|
|
||||||
length: carPropsData?.length || 0,
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return acc;
|
return acc;
|
||||||
}, [] as ICarWagon[]);
|
}, []);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function totalMass(state: IStore) {
|
export function totalWeight(stockList: IStock[]) {
|
||||||
return ~~state.stockList.reduce((acc, stock) => acc + (stock.cargo ? stock.cargo.totalMass : stock.mass) * stock.count, 0);
|
return stockList.reduce(
|
||||||
|
(acc, stock) => acc + (stock.vehicleRef.weight + (stock.cargo?.weight ?? 0)),
|
||||||
|
0
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function totalLength(state: IStore) {
|
export function totalLength(stockList: IStock[]) {
|
||||||
return state.stockList.reduce((acc, stock) => acc + stock.length * stock.count, 0);
|
return stockList.reduce((acc, stock) => acc + stock.vehicleRef.length, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function maxStockSpeed(state: IStore) {
|
export function maxStockSpeed(stockList: IStock[]) {
|
||||||
const stockSpeedLimit = state.stockList.reduce((acc, stock) => (stock.maxSpeed < acc || acc == 0 ? stock.maxSpeed : acc), 0);
|
const stockSpeedLimit = stockList.reduce(
|
||||||
const headingLoco = state.stockList[0]?.isLoco ? state.stockList[0] : undefined;
|
(acc, stock) => (stock.vehicleRef.maxSpeed < acc || acc == 0 ? stock.vehicleRef.maxSpeed : acc),
|
||||||
|
0
|
||||||
|
);
|
||||||
|
const headingLoco =
|
||||||
|
stockList[0] && isTractionUnit(stockList[0].vehicleRef) ? stockList[0] : undefined;
|
||||||
|
|
||||||
if (!headingLoco) return stockSpeedLimit;
|
if (!headingLoco) return stockSpeedLimit;
|
||||||
|
|
||||||
const locoType = headingLoco.type.split('-')[0];
|
const locoType = headingLoco.vehicleRef.type.split('-')[0];
|
||||||
|
|
||||||
if (/^(EN|2EN|SN)/.test(locoType)) return stockSpeedLimit;
|
if (/^(EN|2EN|SN)/.test(locoType)) return stockSpeedLimit;
|
||||||
|
|
||||||
const stockMass = totalMass(state);
|
const speedLimitByMass = calculateSpeedLimit(
|
||||||
|
locoType as SpeedLimitLocoType,
|
||||||
const speedLimitByMass = calculateSpeedLimit(locoType as SpeedLimitLocoType, stockMass, isTrainPassenger(state));
|
totalWeight(stockList),
|
||||||
|
stockList.length,
|
||||||
|
isTrainPassenger(stockList)
|
||||||
|
);
|
||||||
|
|
||||||
return speedLimitByMass ? Math.min(stockSpeedLimit, speedLimitByMass) : stockSpeedLimit;
|
return speedLimitByMass ? Math.min(stockSpeedLimit, speedLimitByMass) : stockSpeedLimit;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function acceptableMass(state: IStore) {
|
export function acceptableWeight(stockList: IStock[]) {
|
||||||
if (state.stockList.length == 0 || !state.stockList[0].isLoco) return 0;
|
if (stockList.length == 0 || !isTractionUnit(stockList[0].vehicleRef)) return 0;
|
||||||
|
|
||||||
const activeLocomotiveType = state.stockList[0].type.split('-')[0];
|
const activeLocomotiveType = stockList[0].vehicleRef.type.split('-')[0];
|
||||||
|
|
||||||
const locoMassLimit = calculateMassLimit(activeLocomotiveType as MassLimitLocoType, isTrainPassenger(state));
|
const locoMassLimit = calculateMassLimit(
|
||||||
|
activeLocomotiveType as MassLimitLocoType,
|
||||||
|
isTrainPassenger(stockList)
|
||||||
|
);
|
||||||
|
|
||||||
return locoMassLimit;
|
return locoMassLimit;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function isTrainPassenger(state: IStore) {
|
export function isTrainPassenger(stockList: IStock[]) {
|
||||||
if (state.stockList.length == 0) return false;
|
if (stockList.length == 0) return false;
|
||||||
if (state.stockList.every((stock) => stock.isLoco)) return false;
|
if (stockList.every((stock) => isTractionUnit(stock.vehicleRef))) return false;
|
||||||
|
|
||||||
return state.stockList.filter((stock) => !stock.isLoco).every((stock) => stock.useType === EVehicleUseType.CAR_PASSENGER);
|
return stockList
|
||||||
}
|
.filter((stock) => !isTractionUnit(stock.vehicleRef))
|
||||||
|
.every((stock) => stock.vehicleRef.group === 'wagon-passenger');
|
||||||
export function chosenRealStock(state: IStore) {
|
|
||||||
const currentStockString = state.stockList
|
|
||||||
.reduce((acc, stock) => {
|
|
||||||
for (let i = 0; i < stock.count; i++) acc.push(stock.type);
|
|
||||||
return acc;
|
|
||||||
}, [] as string[])
|
|
||||||
.join(';');
|
|
||||||
|
|
||||||
const realStockObj = state.readyStockList.find((readyStock) => readyStock.stockString == currentStockString);
|
|
||||||
|
|
||||||
state.chosenRealStockName = realStockObj?.stockId ?? undefined;
|
|
||||||
|
|
||||||
return realStockObj;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,11 +6,11 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent } from "vue";
|
import { defineComponent } from 'vue';
|
||||||
import { useStore } from "../store";
|
import { useStore } from '../store';
|
||||||
|
|
||||||
import MainContainer from "../components/app/MainContainer.vue";
|
import MainContainer from '../components/app/MainContainer.vue';
|
||||||
import FooterVue from "../components/app/Footer.vue";
|
import FooterVue from '../components/app/Footer.vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: {
|
components: {
|
||||||
@@ -26,10 +26,12 @@ export default defineComponent({
|
|||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.app-container {
|
.app-container {
|
||||||
min-height: 100vh;
|
|
||||||
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
justify-content: center;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
|
min-height: 100vh;
|
||||||
|
padding: 0.5em;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
/// <reference types="vite/client" />
|
/// <reference types="vite/client" />
|
||||||
|
|
||||||
declare module "*.vue" {
|
declare module '*.vue' {
|
||||||
import type { DefineComponent } from "vue";
|
import type { DefineComponent } from 'vue';
|
||||||
const component: DefineComponent<{}, {}, any>;
|
const component: DefineComponent<{}, {}, any>;
|
||||||
export default component;
|
export default component;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,13 +6,15 @@ import { VitePWA } from 'vite-plugin-pwa';
|
|||||||
// https://vitejs.dev/config/
|
// https://vitejs.dev/config/
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
server: {
|
server: {
|
||||||
port: 2137,
|
port: 2138,
|
||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
vue(),
|
vue(),
|
||||||
VitePWA({
|
VitePWA({
|
||||||
registerType: 'autoUpdate',
|
registerType: 'autoUpdate',
|
||||||
|
|
||||||
|
includeAssets: ['/images/*.{png,svg,jpg}', '/fonts/*.{woff,woff2,ttf}'],
|
||||||
|
|
||||||
devOptions: {
|
devOptions: {
|
||||||
suppressWarnings: true,
|
suppressWarnings: true,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
@@ -23,8 +25,8 @@ export default defineConfig({
|
|||||||
|
|
||||||
runtimeCaching: [
|
runtimeCaching: [
|
||||||
{
|
{
|
||||||
urlPattern: /^https:\/\/rj.td2.info.pl\/dist\/img\/thumbnails\/.*/i,
|
urlPattern: new RegExp('^https://rj.td2.info.pl/dist/img/thumbnails/*', 'i'),
|
||||||
handler: 'NetworkFirst',
|
handler: 'CacheFirst',
|
||||||
options: {
|
options: {
|
||||||
cacheName: 'swdr-images-cache',
|
cacheName: 'swdr-images-cache',
|
||||||
expiration: {
|
expiration: {
|
||||||
@@ -32,22 +34,16 @@ export default defineConfig({
|
|||||||
maxAgeSeconds: 60 * 60 * 24, // <== 1 day
|
maxAgeSeconds: 60 * 60 * 24, // <== 1 day
|
||||||
},
|
},
|
||||||
cacheableResponse: {
|
cacheableResponse: {
|
||||||
statuses: [0, 200, 404],
|
statuses: [200],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
urlPattern: /^https:\/\/spythere.github.io\/api\/td2\/.*/i,
|
urlPattern: new RegExp('^https://stacjownik.spythere.eu/vehicles', 'i'),
|
||||||
handler: 'NetworkFirst',
|
handler: 'NetworkFirst',
|
||||||
options: {
|
options: {
|
||||||
cacheName: 'spythere-api-cache',
|
cacheName: 'vehicles-cache',
|
||||||
expiration: {
|
|
||||||
maxEntries: 150,
|
|
||||||
maxAgeSeconds: 60 * 60 * 24, // <== 1 day
|
|
||||||
},
|
|
||||||
cacheableResponse: {
|
|
||||||
statuses: [0, 200],
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
|||||||