mirror of
https://github.com/Spythere/pojazdownik.git
synced 2026-05-03 11:45:34 +00:00
Compare commits
20 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| ffce2b572b | |||
| 1b5a26e380 | |||
| c9b681eaee | |||
| 160879adec | |||
| d6ddbe7af7 | |||
| 1c7b72ea1d | |||
| 83f5b07c7e | |||
| 77cb64e25a | |||
| 6d0cc8e7cd | |||
| 7fb4b0ae26 | |||
| ed191d597b | |||
| 5d24accb15 | |||
| 54c6850121 | |||
| fad8c40b4b | |||
| a8485b3531 | |||
| e7dcd125ec | |||
| d886f44c59 | |||
| 5ff6c0504e | |||
| 2158145ae8 | |||
| 6101c9bfb2 |
Generated
+1618
-7294
File diff suppressed because it is too large
Load Diff
+8
-7
@@ -1,7 +1,8 @@
|
|||||||
{
|
{
|
||||||
"name": "pojazdownik",
|
"name": "pojazdownik",
|
||||||
"version": "1.7.3",
|
"version": "1.8.0",
|
||||||
"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
@@ -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);
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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) {
|
||||||
|
|||||||
@@ -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>
|
||||||
@@ -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;
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -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>
|
|
||||||
|
|
||||||
</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 {
|
||||||
|
|||||||
@@ -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 ? `✓` : '✗') : '---' }}
|
{{ isLocomotive(vehicle) ? (vehicle.coldStart ? `✓` : '✗') : '---' }}
|
||||||
</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;
|
||||||
|
|||||||
@@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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,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"]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,17 @@
|
|||||||
},
|
},
|
||||||
"cargo": {
|
"cargo": {
|
||||||
"2000": 70
|
"2000": 70
|
||||||
}
|
},
|
||||||
|
"none": 110
|
||||||
|
},
|
||||||
|
"4E": {
|
||||||
|
"passenger": {
|
||||||
|
"650": 125
|
||||||
|
},
|
||||||
|
"cargo": {
|
||||||
|
"2000": 70
|
||||||
|
},
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"EU07E": {
|
"EU07E": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
@@ -13,25 +23,29 @@
|
|||||||
},
|
},
|
||||||
"cargo": {
|
"cargo": {
|
||||||
"2000": 70
|
"2000": 70
|
||||||
}
|
},
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"EP07": {
|
"EP07": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"650": 125
|
"650": 125
|
||||||
},
|
},
|
||||||
"cargo": null
|
"cargo": null,
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"EP08": {
|
"EP08": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"650": 140
|
"650": 140
|
||||||
},
|
},
|
||||||
"cargo": null
|
"cargo": null,
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"EP09": {
|
"EP09": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
"650": 160
|
"650": 160
|
||||||
},
|
},
|
||||||
"cargo": null
|
"cargo": null,
|
||||||
|
"none": 160
|
||||||
},
|
},
|
||||||
"ET41": {
|
"ET41": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
@@ -39,7 +53,8 @@
|
|||||||
},
|
},
|
||||||
"cargo": {
|
"cargo": {
|
||||||
"4000": 70
|
"4000": 70
|
||||||
}
|
},
|
||||||
|
"none": 110
|
||||||
},
|
},
|
||||||
"SM42": {
|
"SM42": {
|
||||||
"passenger": {
|
"passenger": {
|
||||||
@@ -61,6 +76,7 @@
|
|||||||
"1130": 40,
|
"1130": 40,
|
||||||
"1720": 30,
|
"1720": 30,
|
||||||
"2400": 20
|
"2400": 20
|
||||||
}
|
},
|
||||||
|
"none": 90
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+55
-20
@@ -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
@@ -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
@@ -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");
|
||||||
|
|||||||
@@ -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);
|
||||||
|
|||||||
@@ -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
@@ -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
@@ -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
@@ -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
@@ -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;
|
||||||
|
|||||||
@@ -4,17 +4,18 @@ 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) {
|
||||||
console.log(speedLimits[locoType]);
|
|
||||||
|
|
||||||
if (speedLimits[locoType] === undefined) return 0;
|
if (speedLimits[locoType] === undefined) return 0;
|
||||||
|
|
||||||
const speedTable = speedLimits[locoType][isTrainPassenger ? 'passenger' : 'cargo'];
|
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;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
Reference in New Issue
Block a user