Compare commits

..

29 Commits

Author SHA1 Message Date
Spythere 9ff9341851 Merge pull request #21 from Spythere/development
Wersja 1.8.2
2024-03-08 16:55:16 +01:00
Spythere f9276f6c71 bump: 1.8.2 2024-03-08 16:54:30 +01:00
Spythere f4f9a4729f poprawki cachowania 2024-03-08 16:54:08 +01:00
Spythere ed0906b63e Merge pull request #20 from Spythere/development
Wersja 1.8.1
2024-03-07 14:22:45 +01:00
Spythere 2ae19123a3 bump: 1.8.1 2024-03-07 14:19:23 +01:00
Spythere 0c2be7b927 speed limits fix 2024-03-07 14:19:04 +01:00
Spythere ffce2b572b Wersja 1.7.4
Wersja 1.7.4
2024-02-20 13:26:22 +01:00
Spythere 1b5a26e380 speed constraints for locomotives-only 2024-02-20 12:14:09 +01:00
Spythere c9b681eaee bump: 1.8.0 2024-02-20 11:49:52 +01:00
Spythere 160879adec hotfix 2024-02-19 21:25:21 +01:00
Spythere d6ddbe7af7 translation fixes 2024-02-19 17:42:41 +01:00
Spythere 1c7b72ea1d removed new categories; awaiting changes in the simulator 2024-02-19 17:41:16 +01:00
Spythere 83f5b07c7e code adaptation for API changes; tons changed to kilos 2024-02-19 17:37:46 +01:00
Spythere 77cb64e25a num. generator: categories update 2024-02-15 23:09:21 +01:00
Spythere 6d0cc8e7cd wiki tab td padding 2024-02-14 15:06:45 +01:00
Spythere 7fb4b0ae26 fix: incorrect wagon count while generating a r.stock 2024-02-14 15:04:23 +01:00
Spythere ed191d597b translations; numgen category drawing 2024-02-05 22:36:36 +01:00
Spythere 5d24accb15 forum link update 2024-02-03 18:19:43 +01:00
Spythere 54c6850121 translation & num gen fixes 2024-02-03 18:17:01 +01:00
Spythere fad8c40b4b translation fixes 2024-02-03 16:41:07 +01:00
Spythere a8485b3531 lock sync hotfix 2024-02-03 16:31:04 +01:00
Spythere e7dcd125ec packages upgrade 2024-02-03 16:26:42 +01:00
Spythere d886f44c59 lock files 2024-02-03 16:18:13 +01:00
Spythere 5ff6c0504e bump to 1.7.4; added new categories to number gen & showing rules 2024-02-03 16:15:22 +01:00
Spythere 2158145ae8 design update & fixes 2024-02-01 16:12:41 +01:00
Spythere 6101c9bfb2 grouping generated stock (wip) 2024-01-30 16:39:16 +01:00
Spythere bd9af9a630 Merge pull request #18 from Spythere/development
Wersja 1.7.3
2024-01-18 17:57:01 +01:00
Spythere 2188dbdf9b bump: 1.7.3 2024-01-18 16:37:15 +01:00
Spythere c4576d7802 hotfix 4E 2024-01-18 16:36:57 +01:00
30 changed files with 2338 additions and 7708 deletions
+1618 -7294
View File
File diff suppressed because it is too large Load Diff
+8 -7
View File
@@ -1,7 +1,8 @@
{ {
"name": "pojazdownik", "name": "pojazdownik",
"version": "1.7.2", "version": "1.8.2",
"private": true, "private": true,
"type": "module",
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",
"build": "vue-tsc --noEmit && vite build", "build": "vue-tsc --noEmit && vite build",
@@ -19,17 +20,17 @@
}, },
"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"
} }
} }
+7 -7
View File
@@ -5,11 +5,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 ImageFullscreenPreview from './components/utils/ImageFullscreenPreview.vue'; import ImageFullscreenPreview from "./components/utils/ImageFullscreenPreview.vue";
import AppContainerView from './views/AppContainerView.vue'; 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({
data() { data() {
@@ -26,7 +26,7 @@ export default defineComponent({
</script> </script>
<style lang="scss"> <style lang="scss">
@import './styles/global.scss'; @import "./styles/global.scss";
/* APP */ /* APP */
#app { #app {
@@ -34,7 +34,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);
-5
View File
@@ -41,12 +41,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>
+7 -3
View File
@@ -27,11 +27,15 @@ main {
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;
padding: 1em;
} }
@media screen and (max-width: $breakpointMd) { @media screen and (max-width: $breakpointMd) {
+58
View File
@@ -0,0 +1,58 @@
<template>
<label>
<input type="checkbox" v-model="model" />
<div><slot /></div>
</label>
</template>
<script lang="ts" setup>
const model = defineModel();
</script>
<style lang="scss" scoped>
label {
cursor: pointer;
text-transform: uppercase;
transition: color 200ms;
}
div {
padding: 0.25em 0.5em;
color: white;
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";
}
}
}
</style>
+2 -2
View File
@@ -80,7 +80,7 @@
</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>
@@ -216,7 +216,7 @@ 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;
}); });
}, },
}, },
+20 -15
View File
@@ -23,22 +23,27 @@
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { computed, onMounted, ref } from 'vue'; import { computed, onMounted, ref } from "vue";
import { useStore } from '../../store'; import { useStore } from "../../store";
import StockListTab from '../tabs/StockListTab.vue'; import StockListTab from "../tabs/StockListTab.vue";
import StockGeneratorTab from '../tabs/StockGeneratorTab.vue'; import StockGeneratorTab from "../tabs/StockGeneratorTab.vue";
import NumberGeneratorTab from '../tabs/NumberGeneratorTab.vue'; import NumberGeneratorTab from "../tabs/NumberGeneratorTab.vue";
import WikiListTab from '../tabs/WikiListTab.vue'; import WikiListTab from "../tabs/WikiListTab.vue";
const sectionButtonRefs = ref([]); 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)) {
@@ -51,16 +56,16 @@ onMounted(() => {
const chosenSectionComponent = computed(() => { const chosenSectionComponent = computed(() => {
switch (store.stockSectionMode) { switch (store.stockSectionMode) {
case 'stock-list': case "stock-list":
return StockListTab; return StockListTab;
case 'wiki-list': case "wiki-list":
return WikiListTab; return WikiListTab;
case 'stock-generator': case "stock-generator":
return StockGeneratorTab; return StockGeneratorTab;
case 'number-generator': case "number-generator":
return NumberGeneratorTab; return NumberGeneratorTab;
default: default:
@@ -74,7 +79,7 @@ function chooseSection(sectionId: SectionMode) {
</script> </script>
<style lang="scss"> <style lang="scss">
@import '../../styles/global.scss'; @import "../../styles/global.scss";
// Tab change animation // Tab change animation
.tab-change { .tab-change {
@@ -116,14 +121,14 @@ function chooseSection(sectionId: SectionMode) {
left: 50%; left: 50%;
transform: translateX(-50%); transform: translateX(-50%);
content: ''; content: "";
width: 0; width: 0;
height: 2px; height: 2px;
transition: all 100ms; transition: all 100ms;
background-color: $accentColor; background-color: $accentColor;
} }
&[data-selected='true']::after { &[data-selected="true"]::after {
width: 100%; width: 100%;
} }
} }
@@ -18,7 +18,7 @@
</b> </b>
<div style="color: #ccc"> <div style="color: #ccc">
<div>{{ store.chosenVehicle.length }}m | {{ store.chosenVehicle.mass }}t | {{ store.chosenVehicle.maxSpeed }} km/h</div> <div>{{ store.chosenVehicle.length }}m | {{ (store.chosenVehicle.weight / 1000).toFixed(1) }}t | {{ store.chosenVehicle.maxSpeed }} km/h</div>
<div v-if="isLocomotive(store.chosenVehicle)">{{ $t('preview.cabin') }} {{ store.chosenVehicle.cabinType }}</div> <div v-if="isLocomotive(store.chosenVehicle)">{{ $t('preview.cabin') }} {{ store.chosenVehicle.cabinType }}</div>
+190 -64
View File
@@ -2,46 +2,70 @@
<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="category-select">
<select v-model="chosenCategory" @change="randomizeTrainNumber()"> <label for="category"> {{ $t("numgen.train-category") }}</label>
<select
id="category"
v-model="chosenCategory"
@change="randomizeTrainNumber()"
>
<option :value="null" disabled> <option :value="null" disabled>
{{ $t("numgen.train-category") }} {{ $t("numgen.train-category") }}
</option> </option>
<option <option
v-for="(_, category) in genData.categories" v-for="(_, category) in genData.categoriesRules"
:key="category" :key="category"
:value="category" :value="category"
> >
{{ $t(`numgen.categories.${category}`) }} {{ $t(`numgen.categories.${category}`) }}
</option> </option>
</select> </select>
</div>
<select v-model="beginRegionName" @change="randomizeTrainNumber()"> <div class="regions-select">
<option :value="null" disabled> <div>
{{ $t("numgen.start-region") }} <label for="begin-region"> {{ $t("numgen.start-region") }}</label>
</option> <select
<option id="begin-region"
v-for="(_, name) in genData.regionNumbers" v-model="beginRegionName"
:key="name" @change="randomizeTrainNumber()"
:value="name"
> >
{{ name }} <option :value="null" disabled>
</option> {{ $t("numgen.start-region") }}
</select> </option>
<option
v-for="(_, name) in genData.regionNumbers"
:key="name"
:value="name"
>
{{ name }}
</option>
</select>
</div>
<select v-model="endRegionName" @change="randomizeTrainNumber()"> <div>
<option :value="null" disabled>{{ $t("numgen.end-region") }}</option> <label for="end-region"> {{ $t("numgen.end-region") }}</label>
<option <select
v-for="(_, name) in genData.regionNumbers" id="end-region"
:key="name" v-model="endRegionName"
:value="name" @change="randomizeTrainNumber()"
> >
{{ name }} <option :value="null" disabled>
</option> {{ $t("numgen.end-region") }}
</select> </option>
<option
v-for="(_, name) in genData.regionNumbers"
:key="name"
:value="name"
>
{{ name }}
</option>
</select>
</div>
</div> </div>
<div class="generated-number" @click="copyNumber"> <div class="generated-number" @click="copyNumber">
@@ -52,22 +76,69 @@
<span v-else>{{ $t("numgen.warning") }}</span> <span v-else>{{ $t("numgen.warning") }}</span>
</div> </div>
<!-- <div v-if="chosenCategory"> <div
Current numbering rules: {{ $t(`numgen.rules.${chosenCategory}`) }}; class="category-rules"
v-if="chosenCategory && categoryRules && trainNumber"
>
<!-- 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">
@@ -81,6 +152,11 @@
<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>
@@ -94,10 +170,11 @@ 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>;
@@ -112,6 +189,19 @@ const copyNumber = () => {
} }
}; };
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;
@@ -129,6 +219,7 @@ const randomizeTrainNumber = (randomizeRegions = false) => {
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,32 +238,25 @@ const randomizeTrainNumber = (randomizeRegions = false) => {
return; return;
} }
// Choose default category if it's not chosen
if (chosenCategory.value == null) chosenCategory.value = "EI"; 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))
).toString(); .fill("0")
.join("");
number += number += `${leadingZeros}${randRange}`;
new Array(Math.abs(randRange.length - r.nums)).fill("0").join("") +
randRange;
}
});
trainNumber.value = number; trainNumber.value = number;
}; };
@@ -182,14 +266,39 @@ const randomizeTrainNumber = (randomizeRegions = false) => {
@import "../../styles/tab.scss"; @import "../../styles/tab.scss";
@import "../../styles/global.scss"; @import "../../styles/global.scss";
.options { label {
display: grid; display: block;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr)); font-weight: bold;
margin-bottom: 0.5em;
}
.category-select {
select {
width: auto;
min-width: 50%;
}
margin-bottom: 1em;
}
.regions-select {
display: flex;
align-items: center;
justify-content: center;
gap: 0.5em; gap: 0.5em;
div {
width: 100%;
}
select { select {
width: 100%; width: 100%;
} }
label {
display: block;
margin-bottom: 0.5em;
}
} }
.generated-number { .generated-number {
@@ -204,14 +313,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 +335,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>
+45 -15
View File
@@ -15,7 +15,7 @@
<div class="tab_attributes"> <div class="tab_attributes">
<label> <label>
{{ $t('stockgen.input-mass') }} {{ $t('stockgen.input-mass') }}
<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>
@@ -28,6 +28,12 @@
<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>
@@ -105,7 +111,6 @@ import warningsMixin from '../../mixins/warningsMixin';
export default defineComponent({ export default defineComponent({
name: 'stock-generator', name: 'stock-generator',
mixins: [stockMixin, warningsMixin], mixins: [stockMixin, warningsMixin],
data() { data() {
@@ -117,10 +122,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(),
}; };
}, },
@@ -163,6 +170,15 @@ 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.constructionType + s1.cargo?.id).localeCompare(s2.constructionType + s2.cargo?.id);
});
},
generateStock(empty = false) { generateStock(empty = false) {
const generatedChosenStockList = this.chosenCargoTypes.reduce( const generatedChosenStockList = this.chosenCargoTypes.reduce(
(acc, type) => { (acc, type) => {
@@ -175,8 +191,8 @@ 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) => {
@@ -206,30 +222,29 @@ 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[0]?.isLoco ? 1 : 0);
this.store.stockList.length = headingLoco ? 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 = generatedChosenStockList[~~(Math.random() * generatedChosenStockList.length)]; const randomStockType = 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 +252,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';
}, },
@@ -283,6 +302,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 +347,12 @@ export default defineComponent({
gap: 1em; gap: 1em;
} }
.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;
+21 -79
View File
@@ -33,27 +33,17 @@
</div> </div>
<div class="stock_controls" :data-disabled="store.chosenStockListIndex == -1"> <div class="stock_controls" :data-disabled="store.chosenStockListIndex == -1">
<b v-if="store.chosenStockListIndex >= 0"> <button class="btn btn--image" :tabindex="store.chosenStockListIndex == -1 ? -1 : 0" @click="moveUpStock(store.chosenStockListIndex)">
{{ $t('stocklist.vehicle-no') }}
<span class="text--accent">{{ store.chosenStockListIndex + 1 }}</span>
&nbsp;
</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" /> <img :src="getIconURL('higher')" alt="move up vehicle" />
{{ $t('stocklist.action-move-up') }} {{ $t('stocklist.action-move-up') }}
</button> </button>
<button class="btn" :tabindex="store.chosenStockListIndex == -1 ? -1 : 0" @click="moveDownStock(store.chosenStockListIndex)"> <button class="btn btn--image" :tabindex="store.chosenStockListIndex == -1 ? -1 : 0" @click="moveDownStock(store.chosenStockListIndex)">
<img :src="getIconURL('lower')" alt="move down vehicle" /> <img :src="getIconURL('lower')" alt="move down vehicle" />
{{ $t('stocklist.action-move-down') }} {{ $t('stocklist.action-move-down') }}
</button> </button>
<button class="btn" :tabindex="store.chosenStockListIndex == -1 ? -1 : 0" @click="removeStock(store.chosenStockListIndex)"> <button class="btn btn--image" :tabindex="store.chosenStockListIndex == -1 ? -1 : 0" @click="removeStock(store.chosenStockListIndex)">
<img :src="getIconURL('remove')" alt="remove vehicle" /> <img :src="getIconURL('remove')" alt="remove vehicle" />
{{ $t('stocklist.action-remove') }} {{ $t('stocklist.action-remove') }}
</button> </button>
@@ -70,8 +60,8 @@
<span> <span>
{{ $t('stocklist.mass') }} {{ $t('stocklist.mass') }}
<span class="text--accent">{{ store.totalMass }}t</span> ({{ $t('stocklist.mass-accepted') }}: <span class="text--accent">{{ (store.totalWeight / 1000).toFixed(1) }}t</span> ({{ $t('stocklist.mass-accepted') }}:
<span class="text--accent">{{ store.acceptableMass ? store.acceptableMass + 't' : '-' }}</span <span class="text--accent">{{ store.acceptableWeight ? `${~~(store.acceptableWeight / 1000)}t` : '-' }}</span
>) - {{ $t('stocklist.length') }}: >) - {{ $t('stocklist.length') }}:
<span class="text--accent">{{ store.totalLength }}m</span> <span class="text--accent">{{ store.totalLength }}m</span>
- {{ $t('stocklist.vmax') }}: - {{ $t('stocklist.vmax') }}:
@@ -80,15 +70,13 @@
</div> </div>
<div class="stock_spawn-settings"> <div class="stock_spawn-settings">
<label v-if="store.stockSupportsColdStart" :data-checked="store.isColdStart"> <Checkbox v-if="store.stockSupportsColdStart" v-model="store.isColdStart">
<input type="checkbox" v-model="store.isColdStart" />
{{ $t('stocklist.coldstart-info') }} {{ $t('stocklist.coldstart-info') }}
</label> </Checkbox>
<label v-if="store.stockSupportsDoubleManning" :data-checked="store.isDoubleManned"> <Checkbox v-if="store.stockSupportsDoubleManning" v-model="store.isDoubleManned">
<input type="checkbox" v-model="store.isDoubleManned" />
{{ $t('stocklist.doublemanning-info') }} {{ $t('stocklist.doublemanning-info') }}
</label> </Checkbox>
</div> </div>
<div class="stock_warnings" v-if="stockHasWarnings"> <div class="stock_warnings" v-if="stockHasWarnings">
@@ -122,7 +110,7 @@
<div class="stock-info">{{ $t('stocklist.list-empty') }}</div> <div class="stock-info">{{ $t('stocklist.list-empty') }}</div>
</li> </li>
<TransitionGroup name="stock-list-anim"> <TransitionGroup name="stock-list-anim" v-else>
<li <li
v-for="(stock, i) in store.stockList" v-for="(stock, i) in store.stockList"
:key="stock.id" :key="stock.id"
@@ -148,9 +136,9 @@
<span class="stock-info__cargo" v-if="stock.cargo"> <span class="stock-info__cargo" v-if="stock.cargo">
{{ stock.cargo.id }} {{ stock.cargo.id }}
</span> </span>
<span class="stock-info__length"> {{ stock.length }}m </span> <span class="stock-info__length">{{ stock.length }}m</span>
<span class="stock-info__mass">{{ stock.cargo ? stock.cargo.totalMass : stock.mass }}t </span> <span class="stock-info__mass">{{ ((stock.weight + (stock.cargo?.weight ?? 0)) / 1000).toFixed(1) }}t</span>
<span class="stock-info__speed"> {{ stock.maxSpeed }}km/h </span> <span class="stock-info__speed">{{ stock.maxSpeed }}km/h</span>
</div> </div>
</li> </li>
</TransitionGroup> </TransitionGroup>
@@ -168,10 +156,11 @@ 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';
export default defineComponent({ export default defineComponent({
name: 'stock-list', name: 'stock-list',
components: { StockThumbnails }, components: { StockThumbnails, Checkbox },
mixins: [warningsMixin, imageMixin, stockMixin, stockPreviewMixin], mixins: [warningsMixin, imageMixin, stockMixin, stockPreviewMixin],
@@ -335,7 +324,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.store.chosenRealStockName || this.store.stockList[0].type} ${(this.store.totalWeight / 1000).toFixed(1)}t; ${
this.store.totalLength this.store.totalLength
}m; vmax ${this.store.maxStockSpeed}`; }m; vmax ${this.store.maxStockSpeed}`;
@@ -409,8 +398,9 @@ export default defineComponent({
@import '../../styles/tab.scss'; @import '../../styles/tab.scss';
.stock-list-tab { .stock-list-tab {
display: grid; display: flex;
grid-gap: 0.5em; flex-direction: column;
gap: 0.5em;
} }
.warning { .warning {
@@ -430,9 +420,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 +437,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 +459,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 {
@@ -528,7 +470,7 @@ export default defineComponent({
ul { ul {
position: relative; position: relative;
overflow: auto; overflow: auto;
height: 500px; max-height: 500px;
} }
ul > li { ul > li {
+16 -9
View File
@@ -54,17 +54,23 @@
/> />
</td> </td>
<td :data-sponsoronly="vehicle.isSponsorsOnly">{{ vehicle.type }}</td> <td :data-sponsoronly="vehicle.isSponsorsOnly">
{{ vehicle.type }}
</td>
<td v-if="isLocomotive(vehicle)">{{ $t(`wiki.${vehicle.power}`) }}</td> <td v-if="isLocomotive(vehicle)">
{{ $t(`wiki.${vehicle.power}`) }}
</td>
<td v-else>{{ $t(`wiki.${vehicle.useType}`) }}</td> <td v-else>{{ $t(`wiki.${vehicle.useType}`) }}</td>
<td>{{ vehicle.constructionType }}</td> <td>{{ vehicle.constructionType }}</td>
<td>{{ vehicle.length }}m</td> <td>{{ vehicle.length }}m</td>
<td>{{ vehicle.mass }}t</td> <td>{{ (vehicle.weight / 1000).toFixed(1) }}t</td>
<td>{{ vehicle.maxSpeed }}km/h</td> <td>{{ vehicle.maxSpeed }}km/h</td>
<td v-if="currentFilterMode == 'carriages'">{{ !isLocomotive(vehicle) ? vehicle.cargoList.length : '---' }}</td> <td v-if="currentFilterMode == 'carriages'">
{{ !isLocomotive(vehicle) ? vehicle.cargoTypes.length : '---' }}
</td>
<td v-if="currentFilterMode == 'tractions'"> <td v-if="currentFilterMode == 'tractions'">
{{ isLocomotive(vehicle) ? (vehicle.coldStart ? `&check;` : '&cross;') : '---' }} {{ isLocomotive(vehicle) ? (vehicle.coldStart ? `&check;` : '&cross;') : '---' }}
</td> </td>
@@ -87,7 +93,7 @@ import { isLocomotive } 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'; type SorterID = 'type' | 'constructionType' | 'image' | 'length' | 'weight' | 'maxSpeed' | 'cargoCount' | 'group' | 'coldStart';
interface IWikiHeader { interface IWikiHeader {
id: SorterID; id: SorterID;
@@ -106,7 +112,7 @@ const headers: IWikiHeader[] = [
{ id: 'group', sortable: true, for: 'all' }, { id: 'group', sortable: true, for: 'all' },
{ id: 'constructionType', sortable: true, for: 'all' }, { id: 'constructionType', sortable: true, for: 'all' },
{ id: 'length', sortable: true, for: 'all' }, { id: 'length', sortable: true, for: 'all' },
{ id: 'mass', sortable: true, for: 'all' }, { id: 'weight', sortable: true, for: 'all' },
{ id: 'maxSpeed', sortable: true, for: 'all' }, { id: 'maxSpeed', sortable: true, for: 'all' },
{ id: 'coldStart', sortable: true, for: 'tractions' }, { id: 'coldStart', sortable: true, for: 'tractions' },
{ id: 'cargoCount', sortable: true, for: 'carriages' }, { id: 'cargoCount', sortable: true, for: 'carriages' },
@@ -166,15 +172,15 @@ export default defineComponent({
case 'group': case 'group':
return direction == 1 ? row1.vehicle[id].localeCompare(row2.vehicle[id]) : row2.vehicle[id].localeCompare(row1.vehicle[id]); return direction == 1 ? row1.vehicle[id].localeCompare(row2.vehicle[id]) : row2.vehicle[id].localeCompare(row1.vehicle[id]);
case 'mass': case 'weight':
case 'length': case 'length':
case 'maxSpeed': case 'maxSpeed':
return Math.sign(row1.vehicle[id] - row2.vehicle[id]) * direction; return Math.sign(row1.vehicle[id] - row2.vehicle[id]) * direction;
case 'cargoCount': case 'cargoCount':
return ( return (
(!isLocomotive(row1.vehicle) ? Math.sign(row1.vehicle.cargoList.length || -1) : -1) - (!isLocomotive(row1.vehicle) ? Math.sign(row1.vehicle.cargoTypes.length || -1) : -1) -
(!isLocomotive(row2.vehicle) ? (row2.vehicle.cargoList.length || -1) * direction : -1) (!isLocomotive(row2.vehicle) ? (row2.vehicle.cargoTypes.length || -1) * direction : -1)
); );
case 'coldStart': case 'coldStart':
@@ -290,6 +296,7 @@ export default defineComponent({
td { td {
text-align: center; text-align: center;
height: 70px; height: 70px;
padding: 0.25em;
&[data-sponsoronly='true'] { &[data-sponsoronly='true'] {
color: salmon; color: salmon;
+18 -14
View File
@@ -28,18 +28,18 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { Ref, computed, nextTick, ref, watch } from 'vue'; import { Ref, computed, nextTick, ref, watch } from "vue";
import { useStore } from '../../store'; import { useStore } from "../../store";
import { IStock } from '../../types'; import { IStock } from "../../types";
const store = useStore(); const store = useStore();
const emit = defineEmits(['listItemClick']); const emit = defineEmits(["listItemClick"]);
const thumbnailsRef = ref() as Ref<HTMLElement>; const thumbnailsRef = ref() as Ref<HTMLElement>;
const draggedIndex = ref(-1); const draggedIndex = ref(-1);
const onListItemClick = (index: number) => { const onListItemClick = (index: number) => {
emit('listItemClick', index); emit("listItemClick", index);
}; };
const stockImageError = (e: Event, stock: IStock) => { const stockImageError = (e: Event, stock: IStock) => {
@@ -52,13 +52,15 @@ 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",
});
}); });
} },
); );
// Dragging images // Dragging images
@@ -69,7 +71,9 @@ const onDragStart = (vehicleIndex: number) => {
const onDrop = (e: DragEvent, vehicleIndex: number) => { const onDrop = (e: DragEvent, vehicleIndex: number) => {
e.preventDefault(); e.preventDefault();
let targetEl = thumbnailsRef.value.querySelector(`div:nth-child(${vehicleIndex + 1})`); let targetEl = thumbnailsRef.value.querySelector(
`div:nth-child(${vehicleIndex + 1})`,
);
if (!targetEl && draggedIndex.value != -1) return; if (!targetEl && draggedIndex.value != -1) return;
@@ -110,7 +114,7 @@ const allowDrop = (e: DragEvent) => {
-moz-user-select: none; -moz-user-select: none;
-webkit-user-select: none; -webkit-user-select: none;
&[data-selected='true'] { &[data-selected="true"] {
background-color: rebeccapurple; background-color: rebeccapurple;
} }
@@ -119,7 +123,7 @@ const allowDrop = (e: DragEvent) => {
margin: 0 1em; margin: 0 1em;
} }
&[data-sponsor='true'] > b { &[data-sponsor="true"] > b {
color: salmon; color: salmon;
} }
+9 -8
View File
@@ -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]
} }
+61 -28
View File
@@ -1,37 +1,70 @@
{ {
"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, 66, 60, 69, 77, 70, 79, 88, 80, 89, 92, 98
55, 59, 90, 95, 96, 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", "MP": [null, "050", "169"],
"RO": "2:200-999:3", "RO": [null, "200", "999"],
"PW": "2:6;3:0-899:3", "RP": [null, "050", "169"],
"TM": "2:4;3:0-899:3", "PW": ["6", "000", "899"],
"TK": "2:3;3:0-899:3", "TK": ["3", "000", "899"],
"LT": "2:5;3:0-899:3" "TM": ["4", "000", "899"],
"LT": ["5", "000", "899"]
},
"categoriesNextVersion": {
"EI": [null, "00", "99"],
"EC": [null, "001", "049"],
"EN": [null, "001", "049"],
"RO": [null, "200", "999"],
"RP": [null, "050", "169"],
"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"]
} }
} }
+48 -32
View File
@@ -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
} }
} }
+55 -20
View File
@@ -92,33 +92,68 @@
}, },
"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 service)",
"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": {
@@ -133,7 +168,7 @@
"constructionType": "Construction", "constructionType": "Construction",
"coldStart": "Cold start", "coldStart": "Cold start",
"length": "Length", "length": "Length",
"mass": "Mass", "weight": "Mass",
"maxSpeed": "Speed", "maxSpeed": "Speed",
"cargoCount": "Cargo count" "cargoCount": "Cargo count"
}, },
+55 -20
View File
@@ -92,33 +92,68 @@
}, },
"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": "Początkowy obszar konstrukcyjny",
"end-region": "Końcowy obszar konstrukcyjny", "end-region": "Końcowy obszar konstrukcyjny",
"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ędzywojewódzki pośpieszny",
"RP": "RP - wojewódzki pośpieszny",
"MO": "MO - międzywojewódzki osobowy",
"RO": "RO - wojewódzki osobowy",
"MM": "MM - międzynarodowy pośpieszny",
"MH": "MH - międzywoj. pośpieszny nocny / hotelowy",
"RM": "RM - wojewódzki osobowy międzynarodowy",
"RA": "RA - wojewódzki 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 do obsługi stacji i bocznic",
"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": {
@@ -133,7 +168,7 @@
"constructionType": "Konstrukcja", "constructionType": "Konstrukcja",
"coldStart": "Zimny start", "coldStart": "Zimny start",
"length": "Długość", "length": "Długość",
"mass": "Masa", "weight": "Masa",
"maxSpeed": "Prędkość", "maxSpeed": "Prędkość",
"cargoCount": "Ładunki" "cargoCount": "Ładunki"
}, },
+5 -5
View File
@@ -1,8 +1,8 @@
import { createApp } from 'vue'; import { createApp } from "vue";
import { createPinia } from 'pinia'; import { createPinia } from "pinia";
import App from './App.vue'; import App from "./App.vue";
import i18n from './i18n-setup'; import i18n from "./i18n-setup";
const pinia = createPinia(); const pinia = createPinia();
createApp(App).use(pinia).use(i18n).mount('#app'); createApp(App).use(pinia).use(i18n).mount("#app");
+3 -3
View File
@@ -1,6 +1,6 @@
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, Vehicle } from '../types';
import { isLocomotive } from '../utils/vehicleUtils'; import { isLocomotive } from '../utils/vehicleUtils';
export default defineComponent({ export default defineComponent({
@@ -22,7 +22,7 @@ export default defineComponent({
id: this.getStockId(), id: this.getStockId(),
type: vehicle.type, type: vehicle.type,
length: vehicle.length, length: vehicle.length,
mass: vehicle.mass, weight: vehicle.weight,
maxSpeed: vehicle.maxSpeed, maxSpeed: vehicle.maxSpeed,
isLoco, isLoco,
cargo: !isLoco && vehicle.loadable && cargo ? cargo : undefined, cargo: !isLoco && vehicle.loadable && cargo ? cargo : undefined,
@@ -88,7 +88,7 @@ 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.log('Brak pojazdu / rodzaj pojazdu źle wczytany:', type);
+5 -13
View File
@@ -1,5 +1,5 @@
import { defineComponent } from "vue"; import { defineComponent } from 'vue';
import { useStore } from "../store"; import { useStore } from '../store';
export default defineComponent({ export default defineComponent({
setup() { setup() {
@@ -11,17 +11,11 @@ export default defineComponent({
}, },
computed: { computed: {
trainTooLong() { trainTooLong() {
return ( return (this.store.totalLength > 350 && this.store.isTrainPassenger) || (this.store.totalLength > 650 && !this.store.isTrainPassenger);
(this.store.totalLength > 350 && this.store.isTrainPassenger) ||
(this.store.totalLength > 650 && !this.store.isTrainPassenger)
);
}, },
trainTooHeavy() { trainTooHeavy() {
return ( return this.store.acceptableWeight && this.store.totalWeight > this.store.acceptableWeight;
this.store.acceptableMass &&
this.store.totalMass > this.store.acceptableMass
);
}, },
locoNotSuitable() { locoNotSuitable() {
@@ -29,9 +23,7 @@ export default defineComponent({
!this.store.isTrainPassenger && !this.store.isTrainPassenger &&
this.store.stockList.length > 1 && this.store.stockList.length > 1 &&
!this.store.stockList.every((stock) => stock.isLoco) && !this.store.stockList.every((stock) => stock.isLoco) &&
this.store.stockList.some( this.store.stockList.some((stock) => stock.isLoco && stock.type.startsWith('EP'))
(stock) => stock.isLoco && stock.type.startsWith("EP"),
)
); );
}, },
+4 -4
View File
@@ -1,14 +1,14 @@
import { IStockData, IStore } from './types'; import { IStockData, IStore } from './types';
import { defineStore } from 'pinia'; import { defineStore } from 'pinia';
import { import {
acceptableMass, acceptableMass as acceptableWeight,
carDataList, carDataList,
chosenRealStock, chosenRealStock,
isTrainPassenger, isTrainPassenger,
locoDataList, locoDataList,
maxStockSpeed, maxStockSpeed,
totalLength, totalLength,
totalMass, totalWeight,
} from './utils/vehicleUtils'; } from './utils/vehicleUtils';
import http from './http'; import http from './http';
@@ -56,12 +56,12 @@ export const useStore = defineStore({
locoDataList: (state) => locoDataList(state), locoDataList: (state) => locoDataList(state),
carDataList: (state) => carDataList(state), carDataList: (state) => carDataList(state),
vehicleDataList: (state) => [...locoDataList(state), ...carDataList(state)], vehicleDataList: (state) => [...locoDataList(state), ...carDataList(state)],
totalMass: (state) => totalMass(state), totalWeight: (state) => totalWeight(state),
totalLength: (state) => totalLength(state), totalLength: (state) => totalLength(state),
maxStockSpeed: (state) => maxStockSpeed(state), maxStockSpeed: (state) => maxStockSpeed(state),
isTrainPassenger: (state) => isTrainPassenger(state), isTrainPassenger: (state) => isTrainPassenger(state),
chosenRealStock: (state) => chosenRealStock(state), chosenRealStock: (state) => chosenRealStock(state),
acceptableMass: (state) => acceptableMass(state), acceptableWeight: (state) => acceptableWeight(state),
stockSupportsColdStart: (state) => { stockSupportsColdStart: (state) => {
if (state.stockList.length == 0) return false; if (state.stockList.length == 0) return false;
+22 -15
View File
@@ -3,34 +3,34 @@ $breakpointSm: 550px;
$bgColor: #2b3552; $bgColor: #2b3552;
$textColor: #fff; $textColor: #fff;
$secondaryColor: #222; $secondaryColor: #1b1b1b;
$accentColor: #e4c428; $accentColor: #e4c428;
@font-face { @font-face {
font-family: 'Lato'; font-family: "Lato";
src: src:
url('/fonts/Lato-Light.woff2') format('woff2'), url("/fonts/Lato-Light.woff2") format("woff2"),
url('/fonts/Lato-Light.woff') format('woff'); url("/fonts/Lato-Light.woff") format("woff");
font-weight: 300; font-weight: 300;
font-style: normal; font-style: normal;
font-display: swap; font-display: swap;
} }
@font-face { @font-face {
font-family: 'Lato'; font-family: "Lato";
src: src:
url('/fonts/Lato-Bold.woff2') format('woff2'), url("/fonts/Lato-Bold.woff2") format("woff2"),
url('/fonts/Lato-Bold.woff') format('woff'); url("/fonts/Lato-Bold.woff") format("woff");
font-weight: bold; font-weight: bold;
font-style: normal; font-style: normal;
font-display: swap; font-display: swap;
} }
@font-face { @font-face {
font-family: 'Lato'; font-family: "Lato";
src: src:
url('/fonts/Lato-Regular.woff2') format('woff2'), url("/fonts/Lato-Regular.woff2") format("woff2"),
url('/fonts/Lato-Regular.woff') format('woff'); url("/fonts/Lato-Regular.woff") format("woff");
font-weight: normal; font-weight: normal;
font-style: normal; font-style: normal;
font-display: swap; font-display: swap;
@@ -118,7 +118,7 @@ button {
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;
@@ -141,14 +141,14 @@ button {
outline: 1px solid white; outline: 1px solid white;
} }
&[data-chosen='true'] { &[data-chosen="true"] {
background-color: $accentColor; background-color: $accentColor;
color: black; color: black;
box-shadow: 0 0 5px 1px $accentColor; box-shadow: 0 0 5px 1px $accentColor;
} }
&[data-disabled='true'] { &[data-disabled="true"] {
user-select: none; user-select: none;
pointer-events: none; pointer-events: none;
-moz-user-select: none; -moz-user-select: none;
@@ -183,8 +183,8 @@ button {
} }
select, select,
input[type='text'], input[type="text"],
input[type='number'] { input[type="number"] {
background: $bgColor; background: $bgColor;
border: 2px solid #aaa; border: 2px solid #aaa;
outline: none; outline: none;
@@ -227,6 +227,13 @@ ul {
} }
} }
hr {
height: 3px;
background-color: white;
outline: none;
margin: 0;
}
.g-card { .g-card {
position: fixed; position: fixed;
top: 1em; top: 1em;
+7 -8
View File
@@ -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 {
+15 -14
View File
@@ -41,12 +41,19 @@ export type TCarWagonGroup = 'car-passenger' | 'car-cargo';
export interface IStockProps { export interface IStockProps {
type: string; type: string;
length: number; length: number;
mass: number; // mass: number;
cargo?: string | null; weight: number;
// cargo?: string | null;
cargoTypes: ICargo[] | null;
coldStart?: boolean; coldStart?: boolean;
doubleManned?: boolean; doubleManned?: boolean;
} }
export interface ICargo {
id: string;
weight: number;
}
export interface IStockData { export interface IStockData {
version: string; version: string;
@@ -79,8 +86,7 @@ export interface ILocomotive {
isSponsorsOnly: boolean; isSponsorsOnly: boolean;
sponsorsOnlyTimestamp: number; sponsorsOnlyTimestamp: number;
imageSrc: string; imageSrc: string;
weight: number;
mass: number;
length: number; length: number;
coldStart: boolean; coldStart: boolean;
doubleManned: boolean; doubleManned: boolean;
@@ -96,15 +102,9 @@ export interface ICarWagon {
sponsorsOnlyTimestamp: number; sponsorsOnlyTimestamp: number;
maxSpeed: number; maxSpeed: number;
imageSrc: string; imageSrc: string;
weight: number;
mass: number;
length: number; length: number;
cargoList: { id: string; totalMass: number }[]; cargoTypes: ICargo[];
}
export interface ICargo {
id: string;
totalMass: number;
} }
export interface IStock { export interface IStock {
@@ -113,9 +113,10 @@ export interface IStock {
useType: string; useType: string;
constructionType: string; constructionType: string;
length: number; length: number;
mass: number; // mass: number;
weight: number;
maxSpeed: number; maxSpeed: number;
cargo?: { id: string; totalMass: number }; cargo?: ICargo;
isLoco: boolean; isLoco: boolean;
isSponsorsOnly: boolean; isSponsorsOnly: boolean;
sponsorsOnlyTimestamp: number; sponsorsOnlyTimestamp: number;
+9 -4
View File
@@ -4,19 +4,24 @@ 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(locoType: SpeedLimitLocoType, stockTotalWeight: number, stockCount: number, isTrainPassenger: boolean) {
const speedTable = speedLimits[locoType][isTrainPassenger ? 'passenger' : 'cargo']; 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;
} }
+6 -12
View File
@@ -34,7 +34,7 @@ export function locoDataList(state: IStore) {
imageSrc: '', imageSrc: '',
length: locoProps?.length && type.startsWith('2EN') ? locoProps.length * 2 : locoProps?.length ?? 0, length: locoProps?.length && type.startsWith('2EN') ? locoProps.length * 2 : locoProps?.length ?? 0,
mass: locoProps?.mass && type.startsWith('2EN') ? 253 : locoProps?.mass ?? 0, weight: locoProps?.weight && type.startsWith('2EN') ? 253000 : locoProps?.weight ?? 0,
coldStart: locoProps?.coldStart ?? false, coldStart: locoProps?.coldStart ?? false,
doubleManned: locoProps?.doubleManned ?? false, doubleManned: locoProps?.doubleManned ?? false,
@@ -72,13 +72,9 @@ export function carDataList(state: IStore) {
sponsorsOnlyTimestamp: Number(sponsorsOnlyTimestamp), sponsorsOnlyTimestamp: Number(sponsorsOnlyTimestamp),
maxSpeed: Number(maxSpeed), maxSpeed: Number(maxSpeed),
imageSrc: '', imageSrc: '',
cargoList: cargoTypes: carPropsData?.cargoTypes ?? [],
carPropsData?.cargo?.split(';').map((cargo) => ({
id: cargo.split(':')[0],
totalMass: Number(cargo.split(':')[1]),
})) || [],
mass: carPropsData?.mass || 0, weight: carPropsData?.weight || 0,
length: carPropsData?.length || 0, length: carPropsData?.length || 0,
}); });
}); });
@@ -87,8 +83,8 @@ export function carDataList(state: IStore) {
}, [] as ICarWagon[]); }, [] as ICarWagon[]);
} }
export function totalMass(state: IStore) { export function totalWeight(state: IStore) {
return ~~state.stockList.reduce((acc, stock) => acc + (stock.cargo ? stock.cargo.totalMass : stock.mass) * stock.count, 0); return state.stockList.reduce((acc, stock) => acc + (stock.weight + (stock.cargo?.weight ?? 0)) * stock.count, 0);
} }
export function totalLength(state: IStore) { export function totalLength(state: IStore) {
@@ -105,9 +101,7 @@ export function maxStockSpeed(state: IStore) {
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, totalWeight(state), state.stockList.length, isTrainPassenger(state));
const speedLimitByMass = calculateSpeedLimit(locoType as SpeedLimitLocoType, stockMass, isTrainPassenger(state));
return speedLimitByMass ? Math.min(stockSpeedLimit, speedLimitByMass) : stockSpeedLimit; return speedLimitByMass ? Math.min(stockSpeedLimit, speedLimitByMass) : stockSpeedLimit;
} }
+5 -3
View File
@@ -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>
+17 -3
View File
@@ -24,7 +24,7 @@ export default defineConfig({
runtimeCaching: [ runtimeCaching: [
{ {
urlPattern: /^https:\/\/rj.td2.info.pl\/dist\/img\/thumbnails\/.*/i, urlPattern: /^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: {
@@ -37,10 +37,24 @@ export default defineConfig({
}, },
}, },
{ {
urlPattern: /^https:\/\/spythere.github.io\/api\/td2\/.*/i, urlPattern: /^https:\/\/spythere.github.io\/api\/td2\/data\/.*/i,
handler: 'NetworkFirst', handler: 'NetworkFirst',
options: { options: {
cacheName: 'spythere-api-cache', cacheName: 'spythere-api-data-cache',
expiration: {
maxEntries: 150,
maxAgeSeconds: 60 * 60 * 24, // <== 1 day
},
cacheableResponse: {
statuses: [0, 200],
},
},
},
{
urlPattern: /^https:\/\/spythere.github.io\/api\/td2\/images\/.*/i,
handler: 'CacheFirst',
options: {
cacheName: 'spythere-api-images-cache',
expiration: { expiration: {
maxEntries: 150, maxEntries: 150,
maxAgeSeconds: 60 * 60 * 24, // <== 1 day maxAgeSeconds: 60 * 60 * 24, // <== 1 day