Compare commits
219 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| caf6137ca3 | |||
| 0f2e5e084b | |||
| fc7a9be9dd | |||
| c0b892da97 | |||
| 907b75f12b | |||
| 3c3a114a38 | |||
| 47f824bef0 | |||
| 2d3e830cf9 | |||
| c888b3d386 | |||
| 645a70ef9c | |||
| 1cd93f09c4 | |||
| 6b4231496e | |||
| b72ee13bdb | |||
| ce053a5a82 | |||
| b08e39ae1a | |||
| dc27500237 | |||
| fe6972c1f8 | |||
| 47193181e5 | |||
| 08b9b72dcd | |||
| 7bbabdd7bf | |||
| c90be042e7 | |||
| 200318def7 | |||
| 430a05ab38 | |||
| f335ca8fc2 | |||
| 15e599fe3c | |||
| bd25914ed4 | |||
| 01ea259381 | |||
| aea26fa538 | |||
| 28d78cd2bc | |||
| a021deae96 | |||
| 8840576796 | |||
| 5018e21736 | |||
| a7fa1dfb6d | |||
| a3558c0b30 | |||
| ee159fd582 | |||
| 35c9fb7ef1 | |||
| e24097c240 | |||
| 01cbebd019 | |||
| 3a5ef7e025 | |||
| c78a5b4d67 | |||
| 023de9f7b8 | |||
| 1024e44cc0 | |||
| 580d404d4a | |||
| 6d1ef26ac1 | |||
| bf9799e0c3 | |||
| 1d13e31d79 | |||
| 16f272bd7d | |||
| 23ca33264c | |||
| 324ca3de4d | |||
| e0548e593c | |||
| 2727350837 | |||
| 6c3af0a8d3 | |||
| e784202a36 | |||
| c24f691693 | |||
| 3aeabd63c9 | |||
| 4c79376318 | |||
| bc1c446c37 | |||
| fba335d0c7 | |||
| b4e536da40 | |||
| 8cde8e6323 | |||
| d7a9e93978 | |||
| 69aa62e77f | |||
| 4b842627fb | |||
| 5ffc63a815 | |||
| 87f7ff58e8 | |||
| 8b6944a8e5 | |||
| cfeeb8fddd | |||
| 89f7fd3c53 | |||
| 86259988c9 | |||
| 7b5ef18ad6 | |||
| d784042691 | |||
| d0e482aa4f | |||
| 3bf1db52b4 | |||
| 8e713a5c6e | |||
| af6eb35b67 | |||
| 1e6ab1c2d1 | |||
| fd4849bd5e | |||
| bc0f4c5d3f | |||
| 8909a0cd40 | |||
| a2602aeefe | |||
| 37ad9b2787 | |||
| 0b4ad679b3 | |||
| dd0d7897cf | |||
| 1453dbda01 | |||
| 4af856b833 | |||
| 182b46a377 | |||
| bb5fc395d2 | |||
| a91a00f88a | |||
| c8d481a952 | |||
| 03ff4d8648 | |||
| 23767801d5 | |||
| 310261fb59 | |||
| 742754ceef | |||
| b9bb9dc201 | |||
| 611927f96f | |||
| 2e191f355e | |||
| f974643e37 | |||
| 02afe2bf33 | |||
| ebdffc6241 | |||
| 911c051af3 | |||
| 7ab16960ca | |||
| 43c7b8b024 | |||
| 32cf7745e8 | |||
| a68c5020d9 | |||
| bf88caa704 | |||
| ea5d681943 | |||
| d346e049e0 | |||
| feb2027c16 | |||
| 06d0fabc99 | |||
| 3c74580bed | |||
| 02d234a21b | |||
| d9dc44063f | |||
| 5929bbdccb | |||
| f73c3f4aec | |||
| a44ad5c89d | |||
| c30c2206ce | |||
| 128f3c32b4 | |||
| 0fdcd82754 | |||
| 1e75ff517f | |||
| b278c20480 | |||
| fd28eb4609 | |||
| a602358241 | |||
| 5a09543a22 | |||
| f952a7c491 | |||
| adf4d88cb2 | |||
| 34f2a69863 | |||
| b2930f6a9e | |||
| edcaff2183 | |||
| 010ab08701 | |||
| 16b3bb3683 | |||
| 93e242c0f5 | |||
| 861206a5ab | |||
| a47399fe1b | |||
| 8e196c8279 | |||
| be55bac9fe | |||
| c5e53057eb | |||
| 4ba5d544af | |||
| 22b6177560 | |||
| 9b6c6ee756 | |||
| 829059d35b | |||
| b56e114ef9 | |||
| 71b4cc3bdb | |||
| 8cc773ffb5 | |||
| 427b4c03e4 | |||
| 46dc43d652 | |||
| 6435d12090 | |||
| e41b8cfa98 | |||
| bc81bb2a38 | |||
| e6c064d15d | |||
| 4d1df5165c | |||
| 43ac2be3e7 | |||
| 75c4e56183 | |||
| 931f6b9fbd | |||
| 21fa1f8699 | |||
| 877ef50a97 | |||
| 933be53630 | |||
| eef4103960 | |||
| 9fafbe2c7f | |||
| 666ba07307 | |||
| b63328f97c | |||
| 342127d541 | |||
| c6ab0d21de | |||
| da4476bdf0 | |||
| a950b4bef4 | |||
| 5aa9297ec5 | |||
| 0af49befc6 | |||
| 4da0ab475b | |||
| 1fa5934784 | |||
| 5fb1a87b41 | |||
| 8a5687cc01 | |||
| c5fe929b9a | |||
| 5787deeaf8 | |||
| 130732921b | |||
| 1b2cd34e86 | |||
| 17bda9e6e7 | |||
| c66ff8feed | |||
| 027cdee25a | |||
| 435cfb3b3f | |||
| 425241c8e7 | |||
| f24f961d52 | |||
| 4718eeeaaf | |||
| 931fd7b21b | |||
| bb79c5033a | |||
| ee290788dc | |||
| a87d1060d3 | |||
| 1804d6d0f0 | |||
| 77250e30c7 | |||
| c5aefd03b8 | |||
| 2ec4694bd3 | |||
| 729f66bcdb | |||
| b746843086 | |||
| cbbd06fecd | |||
| 11e99b6af0 | |||
| 31f4a2e5b2 | |||
| 22514c3263 | |||
| 0df673467c | |||
| 6377e13809 | |||
| 13fa633db4 | |||
| dd9661551c | |||
| 495012a5ca | |||
| 3cfccb1bb4 | |||
| d2a8cdb2b0 | |||
| c33b5ef8c1 | |||
| 52d1771c21 | |||
| cac4345683 | |||
| 6fd9e21213 | |||
| 421add1ec1 | |||
| 4ac92198b7 | |||
| 5105229eef | |||
| 2ec02080c3 | |||
| 95b2a696e1 | |||
| 091e94e396 | |||
| 43c939bf01 | |||
| 8285d5c579 | |||
| 547248b478 | |||
| e1b9b37ac8 | |||
| c8ec28292b | |||
| 3daf800a89 | |||
| 69d9be0bb3 |
@@ -1,17 +0,0 @@
|
|||||||
on:
|
|
||||||
release:
|
|
||||||
types: [published]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
github-releases-to-discord:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v3
|
|
||||||
- name: Github Releases To Discord
|
|
||||||
uses: SethCohen/github-releases-to-discord@v1.13.1
|
|
||||||
with:
|
|
||||||
webhook_url: ${{ secrets.WEBHOOK_URL }}
|
|
||||||
color: "15844367"
|
|
||||||
footer_title: "Changelog - Stacjownik"
|
|
||||||
footer_timestamp: true
|
|
||||||
@@ -33,6 +33,10 @@ node_modules
|
|||||||
|
|
||||||
# Env
|
# Env
|
||||||
.env
|
.env
|
||||||
|
.env.*
|
||||||
|
|
||||||
.fake
|
.fake
|
||||||
.ionide
|
.ionide
|
||||||
|
|
||||||
|
# api-mock
|
||||||
|
/api-mock/endpoints/
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
import { existsSync } from 'fs';
|
||||||
|
import { mkdir, writeFile } from 'fs/promises';
|
||||||
|
|
||||||
|
async function fetchJSONEndpointData(url, fileName) {
|
||||||
|
try {
|
||||||
|
const res = await fetch(url);
|
||||||
|
const data = await res.json();
|
||||||
|
await writeFile(`./endpoints/${fileName}`, JSON.stringify(data));
|
||||||
|
|
||||||
|
return true;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function main() {
|
||||||
|
if (!existsSync('endpoints')) await mkdir('endpoints');
|
||||||
|
|
||||||
|
Promise.all(
|
||||||
|
['getActiveData', 'getDonators', 'getSceneries', 'getVehicles'].map((endpointName) =>
|
||||||
|
fetchJSONEndpointData(
|
||||||
|
`https://stacjownik.spythere.eu/api/${endpointName}`,
|
||||||
|
`${endpointName}.json`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
).then(() => {
|
||||||
|
console.log('Endpoints downloaded!');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
main();
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
import express from 'express';
|
||||||
|
import path from 'path';
|
||||||
|
import { cwd } from 'process';
|
||||||
|
import cors from 'cors';
|
||||||
|
|
||||||
|
const app = express();
|
||||||
|
|
||||||
|
app.use(cors());
|
||||||
|
|
||||||
|
app.get('/api/getActiveData', (_, res) => {
|
||||||
|
res.sendFile(path.join(cwd(), 'endpoints', 'getActiveData.json'));
|
||||||
|
});
|
||||||
|
|
||||||
|
app.get('/api/getSceneries', (_, res) => {
|
||||||
|
res.sendFile(path.join(cwd(), 'endpoints', 'getSceneries.json'));
|
||||||
|
});
|
||||||
|
|
||||||
|
app.get('/api/getVehicles', (_, res) => {
|
||||||
|
res.sendFile(path.join(cwd(), 'endpoints', 'getVehicles.json'));
|
||||||
|
});
|
||||||
|
|
||||||
|
app.get('/api/getDonators', (_, res) => {
|
||||||
|
res.sendFile(path.join(cwd(), 'endpoints', 'getDonators.json'));
|
||||||
|
});
|
||||||
|
|
||||||
|
app.listen(3123, () => {
|
||||||
|
console.log('Mocking API server...');
|
||||||
|
});
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
{
|
||||||
|
"name": "api-mock",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "",
|
||||||
|
"type": "module",
|
||||||
|
"main": "index.js",
|
||||||
|
"scripts": {
|
||||||
|
"test": "echo \"Error: no test specified\" && exit 1",
|
||||||
|
"start": "node index.js",
|
||||||
|
"fetch": "node fetchEndpoints.js"
|
||||||
|
},
|
||||||
|
"author": "",
|
||||||
|
"license": "ISC",
|
||||||
|
"dependencies": {
|
||||||
|
"cors": "^2.8.5",
|
||||||
|
"express": "^4.18.3"
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,481 @@
|
|||||||
|
# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
|
||||||
|
# yarn lockfile v1
|
||||||
|
|
||||||
|
|
||||||
|
accepts@~1.3.8:
|
||||||
|
version "1.3.8"
|
||||||
|
resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.8.tgz#0bf0be125b67014adcb0b0921e62db7bffe16b2e"
|
||||||
|
integrity sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==
|
||||||
|
dependencies:
|
||||||
|
mime-types "~2.1.34"
|
||||||
|
negotiator "0.6.3"
|
||||||
|
|
||||||
|
array-flatten@1.1.1:
|
||||||
|
version "1.1.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2"
|
||||||
|
integrity sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg==
|
||||||
|
|
||||||
|
body-parser@1.20.3:
|
||||||
|
version "1.20.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.20.3.tgz#1953431221c6fb5cd63c4b36d53fab0928e548c6"
|
||||||
|
integrity sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g==
|
||||||
|
dependencies:
|
||||||
|
bytes "3.1.2"
|
||||||
|
content-type "~1.0.5"
|
||||||
|
debug "2.6.9"
|
||||||
|
depd "2.0.0"
|
||||||
|
destroy "1.2.0"
|
||||||
|
http-errors "2.0.0"
|
||||||
|
iconv-lite "0.4.24"
|
||||||
|
on-finished "2.4.1"
|
||||||
|
qs "6.13.0"
|
||||||
|
raw-body "2.5.2"
|
||||||
|
type-is "~1.6.18"
|
||||||
|
unpipe "1.0.0"
|
||||||
|
|
||||||
|
bytes@3.1.2:
|
||||||
|
version "3.1.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.2.tgz#8b0beeb98605adf1b128fa4386403c009e0221a5"
|
||||||
|
integrity sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==
|
||||||
|
|
||||||
|
call-bind@^1.0.7:
|
||||||
|
version "1.0.7"
|
||||||
|
resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.7.tgz#06016599c40c56498c18769d2730be242b6fa3b9"
|
||||||
|
integrity sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==
|
||||||
|
dependencies:
|
||||||
|
es-define-property "^1.0.0"
|
||||||
|
es-errors "^1.3.0"
|
||||||
|
function-bind "^1.1.2"
|
||||||
|
get-intrinsic "^1.2.4"
|
||||||
|
set-function-length "^1.2.1"
|
||||||
|
|
||||||
|
content-disposition@0.5.4:
|
||||||
|
version "0.5.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.4.tgz#8b82b4efac82512a02bb0b1dcec9d2c5e8eb5bfe"
|
||||||
|
integrity sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ==
|
||||||
|
dependencies:
|
||||||
|
safe-buffer "5.2.1"
|
||||||
|
|
||||||
|
content-type@~1.0.4, content-type@~1.0.5:
|
||||||
|
version "1.0.5"
|
||||||
|
resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.5.tgz#8b773162656d1d1086784c8f23a54ce6d73d7918"
|
||||||
|
integrity sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==
|
||||||
|
|
||||||
|
cookie-signature@1.0.6:
|
||||||
|
version "1.0.6"
|
||||||
|
resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c"
|
||||||
|
integrity sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ==
|
||||||
|
|
||||||
|
cookie@0.6.0:
|
||||||
|
version "0.6.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.6.0.tgz#2798b04b071b0ecbff0dbb62a505a8efa4e19051"
|
||||||
|
integrity sha512-U71cyTamuh1CRNCfpGY6to28lxvNwPG4Guz/EVjgf3Jmzv0vlDp1atT9eS5dDjMYHucpHbWns6Lwf3BKz6svdw==
|
||||||
|
|
||||||
|
cors@^2.8.5:
|
||||||
|
version "2.8.5"
|
||||||
|
resolved "https://registry.yarnpkg.com/cors/-/cors-2.8.5.tgz#eac11da51592dd86b9f06f6e7ac293b3df875d29"
|
||||||
|
integrity sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==
|
||||||
|
dependencies:
|
||||||
|
object-assign "^4"
|
||||||
|
vary "^1"
|
||||||
|
|
||||||
|
debug@2.6.9:
|
||||||
|
version "2.6.9"
|
||||||
|
resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f"
|
||||||
|
integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==
|
||||||
|
dependencies:
|
||||||
|
ms "2.0.0"
|
||||||
|
|
||||||
|
define-data-property@^1.1.4:
|
||||||
|
version "1.1.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/define-data-property/-/define-data-property-1.1.4.tgz#894dc141bb7d3060ae4366f6a0107e68fbe48c5e"
|
||||||
|
integrity sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==
|
||||||
|
dependencies:
|
||||||
|
es-define-property "^1.0.0"
|
||||||
|
es-errors "^1.3.0"
|
||||||
|
gopd "^1.0.1"
|
||||||
|
|
||||||
|
depd@2.0.0:
|
||||||
|
version "2.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/depd/-/depd-2.0.0.tgz#b696163cc757560d09cf22cc8fad1571b79e76df"
|
||||||
|
integrity sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==
|
||||||
|
|
||||||
|
destroy@1.2.0:
|
||||||
|
version "1.2.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.2.0.tgz#4803735509ad8be552934c67df614f94e66fa015"
|
||||||
|
integrity sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg==
|
||||||
|
|
||||||
|
ee-first@1.1.1:
|
||||||
|
version "1.1.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d"
|
||||||
|
integrity sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==
|
||||||
|
|
||||||
|
encodeurl@~1.0.2:
|
||||||
|
version "1.0.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59"
|
||||||
|
integrity sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==
|
||||||
|
|
||||||
|
encodeurl@~2.0.0:
|
||||||
|
version "2.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-2.0.0.tgz#7b8ea898077d7e409d3ac45474ea38eaf0857a58"
|
||||||
|
integrity sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg==
|
||||||
|
|
||||||
|
es-define-property@^1.0.0:
|
||||||
|
version "1.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/es-define-property/-/es-define-property-1.0.0.tgz#c7faefbdff8b2696cf5f46921edfb77cc4ba3845"
|
||||||
|
integrity sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==
|
||||||
|
dependencies:
|
||||||
|
get-intrinsic "^1.2.4"
|
||||||
|
|
||||||
|
es-errors@^1.3.0:
|
||||||
|
version "1.3.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/es-errors/-/es-errors-1.3.0.tgz#05f75a25dab98e4fb1dcd5e1472c0546d5057c8f"
|
||||||
|
integrity sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==
|
||||||
|
|
||||||
|
escape-html@~1.0.3:
|
||||||
|
version "1.0.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988"
|
||||||
|
integrity sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==
|
||||||
|
|
||||||
|
etag@~1.8.1:
|
||||||
|
version "1.8.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887"
|
||||||
|
integrity sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==
|
||||||
|
|
||||||
|
express@^4.18.3:
|
||||||
|
version "4.21.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/express/-/express-4.21.0.tgz#d57cb706d49623d4ac27833f1cbc466b668eb915"
|
||||||
|
integrity sha512-VqcNGcj/Id5ZT1LZ/cfihi3ttTn+NJmkli2eZADigjq29qTlWi/hAQ43t/VLPq8+UX06FCEx3ByOYet6ZFblng==
|
||||||
|
dependencies:
|
||||||
|
accepts "~1.3.8"
|
||||||
|
array-flatten "1.1.1"
|
||||||
|
body-parser "1.20.3"
|
||||||
|
content-disposition "0.5.4"
|
||||||
|
content-type "~1.0.4"
|
||||||
|
cookie "0.6.0"
|
||||||
|
cookie-signature "1.0.6"
|
||||||
|
debug "2.6.9"
|
||||||
|
depd "2.0.0"
|
||||||
|
encodeurl "~2.0.0"
|
||||||
|
escape-html "~1.0.3"
|
||||||
|
etag "~1.8.1"
|
||||||
|
finalhandler "1.3.1"
|
||||||
|
fresh "0.5.2"
|
||||||
|
http-errors "2.0.0"
|
||||||
|
merge-descriptors "1.0.3"
|
||||||
|
methods "~1.1.2"
|
||||||
|
on-finished "2.4.1"
|
||||||
|
parseurl "~1.3.3"
|
||||||
|
path-to-regexp "0.1.10"
|
||||||
|
proxy-addr "~2.0.7"
|
||||||
|
qs "6.13.0"
|
||||||
|
range-parser "~1.2.1"
|
||||||
|
safe-buffer "5.2.1"
|
||||||
|
send "0.19.0"
|
||||||
|
serve-static "1.16.2"
|
||||||
|
setprototypeof "1.2.0"
|
||||||
|
statuses "2.0.1"
|
||||||
|
type-is "~1.6.18"
|
||||||
|
utils-merge "1.0.1"
|
||||||
|
vary "~1.1.2"
|
||||||
|
|
||||||
|
finalhandler@1.3.1:
|
||||||
|
version "1.3.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.3.1.tgz#0c575f1d1d324ddd1da35ad7ece3df7d19088019"
|
||||||
|
integrity sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ==
|
||||||
|
dependencies:
|
||||||
|
debug "2.6.9"
|
||||||
|
encodeurl "~2.0.0"
|
||||||
|
escape-html "~1.0.3"
|
||||||
|
on-finished "2.4.1"
|
||||||
|
parseurl "~1.3.3"
|
||||||
|
statuses "2.0.1"
|
||||||
|
unpipe "~1.0.0"
|
||||||
|
|
||||||
|
forwarded@0.2.0:
|
||||||
|
version "0.2.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.2.0.tgz#2269936428aad4c15c7ebe9779a84bf0b2a81811"
|
||||||
|
integrity sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==
|
||||||
|
|
||||||
|
fresh@0.5.2:
|
||||||
|
version "0.5.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.2.tgz#3d8cadd90d976569fa835ab1f8e4b23a105605a7"
|
||||||
|
integrity sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==
|
||||||
|
|
||||||
|
function-bind@^1.1.2:
|
||||||
|
version "1.1.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.2.tgz#2c02d864d97f3ea6c8830c464cbd11ab6eab7a1c"
|
||||||
|
integrity sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==
|
||||||
|
|
||||||
|
get-intrinsic@^1.1.3, get-intrinsic@^1.2.4:
|
||||||
|
version "1.2.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.2.4.tgz#e385f5a4b5227d449c3eabbad05494ef0abbeadd"
|
||||||
|
integrity sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==
|
||||||
|
dependencies:
|
||||||
|
es-errors "^1.3.0"
|
||||||
|
function-bind "^1.1.2"
|
||||||
|
has-proto "^1.0.1"
|
||||||
|
has-symbols "^1.0.3"
|
||||||
|
hasown "^2.0.0"
|
||||||
|
|
||||||
|
gopd@^1.0.1:
|
||||||
|
version "1.0.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.0.1.tgz#29ff76de69dac7489b7c0918a5788e56477c332c"
|
||||||
|
integrity sha512-d65bNlIadxvpb/A2abVdlqKqV563juRnZ1Wtk6s1sIR8uNsXR70xqIzVqxVf1eTqDunwT2MkczEeaezCKTZhwA==
|
||||||
|
dependencies:
|
||||||
|
get-intrinsic "^1.1.3"
|
||||||
|
|
||||||
|
has-property-descriptors@^1.0.2:
|
||||||
|
version "1.0.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz#963ed7d071dc7bf5f084c5bfbe0d1b6222586854"
|
||||||
|
integrity sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==
|
||||||
|
dependencies:
|
||||||
|
es-define-property "^1.0.0"
|
||||||
|
|
||||||
|
has-proto@^1.0.1:
|
||||||
|
version "1.0.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/has-proto/-/has-proto-1.0.3.tgz#b31ddfe9b0e6e9914536a6ab286426d0214f77fd"
|
||||||
|
integrity sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==
|
||||||
|
|
||||||
|
has-symbols@^1.0.3:
|
||||||
|
version "1.0.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.0.3.tgz#bb7b2c4349251dce87b125f7bdf874aa7c8b39f8"
|
||||||
|
integrity sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==
|
||||||
|
|
||||||
|
hasown@^2.0.0:
|
||||||
|
version "2.0.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.2.tgz#003eaf91be7adc372e84ec59dc37252cedb80003"
|
||||||
|
integrity sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==
|
||||||
|
dependencies:
|
||||||
|
function-bind "^1.1.2"
|
||||||
|
|
||||||
|
http-errors@2.0.0:
|
||||||
|
version "2.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-2.0.0.tgz#b7774a1486ef73cf7667ac9ae0858c012c57b9d3"
|
||||||
|
integrity sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==
|
||||||
|
dependencies:
|
||||||
|
depd "2.0.0"
|
||||||
|
inherits "2.0.4"
|
||||||
|
setprototypeof "1.2.0"
|
||||||
|
statuses "2.0.1"
|
||||||
|
toidentifier "1.0.1"
|
||||||
|
|
||||||
|
iconv-lite@0.4.24:
|
||||||
|
version "0.4.24"
|
||||||
|
resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b"
|
||||||
|
integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==
|
||||||
|
dependencies:
|
||||||
|
safer-buffer ">= 2.1.2 < 3"
|
||||||
|
|
||||||
|
inherits@2.0.4:
|
||||||
|
version "2.0.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
|
||||||
|
integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
|
||||||
|
|
||||||
|
ipaddr.js@1.9.1:
|
||||||
|
version "1.9.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz#bff38543eeb8984825079ff3a2a8e6cbd46781b3"
|
||||||
|
integrity sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==
|
||||||
|
|
||||||
|
media-typer@0.3.0:
|
||||||
|
version "0.3.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748"
|
||||||
|
integrity sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==
|
||||||
|
|
||||||
|
merge-descriptors@1.0.3:
|
||||||
|
version "1.0.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.3.tgz#d80319a65f3c7935351e5cfdac8f9318504dbed5"
|
||||||
|
integrity sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ==
|
||||||
|
|
||||||
|
methods@~1.1.2:
|
||||||
|
version "1.1.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee"
|
||||||
|
integrity sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==
|
||||||
|
|
||||||
|
mime-db@1.52.0:
|
||||||
|
version "1.52.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70"
|
||||||
|
integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==
|
||||||
|
|
||||||
|
mime-types@~2.1.24, mime-types@~2.1.34:
|
||||||
|
version "2.1.35"
|
||||||
|
resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a"
|
||||||
|
integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==
|
||||||
|
dependencies:
|
||||||
|
mime-db "1.52.0"
|
||||||
|
|
||||||
|
mime@1.6.0:
|
||||||
|
version "1.6.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/mime/-/mime-1.6.0.tgz#32cd9e5c64553bd58d19a568af452acff04981b1"
|
||||||
|
integrity sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==
|
||||||
|
|
||||||
|
ms@2.0.0:
|
||||||
|
version "2.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8"
|
||||||
|
integrity sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==
|
||||||
|
|
||||||
|
ms@2.1.3:
|
||||||
|
version "2.1.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2"
|
||||||
|
integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==
|
||||||
|
|
||||||
|
negotiator@0.6.3:
|
||||||
|
version "0.6.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.3.tgz#58e323a72fedc0d6f9cd4d31fe49f51479590ccd"
|
||||||
|
integrity sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==
|
||||||
|
|
||||||
|
object-assign@^4:
|
||||||
|
version "4.1.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863"
|
||||||
|
integrity sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==
|
||||||
|
|
||||||
|
object-inspect@^1.13.1:
|
||||||
|
version "1.13.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.13.2.tgz#dea0088467fb991e67af4058147a24824a3043ff"
|
||||||
|
integrity sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==
|
||||||
|
|
||||||
|
on-finished@2.4.1:
|
||||||
|
version "2.4.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.4.1.tgz#58c8c44116e54845ad57f14ab10b03533184ac3f"
|
||||||
|
integrity sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==
|
||||||
|
dependencies:
|
||||||
|
ee-first "1.1.1"
|
||||||
|
|
||||||
|
parseurl@~1.3.3:
|
||||||
|
version "1.3.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.3.tgz#9da19e7bee8d12dff0513ed5b76957793bc2e8d4"
|
||||||
|
integrity sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==
|
||||||
|
|
||||||
|
path-to-regexp@0.1.10:
|
||||||
|
version "0.1.10"
|
||||||
|
resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-0.1.10.tgz#67e9108c5c0551b9e5326064387de4763c4d5f8b"
|
||||||
|
integrity sha512-7lf7qcQidTku0Gu3YDPc8DJ1q7OOucfa/BSsIwjuh56VU7katFvuM8hULfkwB3Fns/rsVF7PwPKVw1sl5KQS9w==
|
||||||
|
|
||||||
|
proxy-addr@~2.0.7:
|
||||||
|
version "2.0.7"
|
||||||
|
resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-2.0.7.tgz#f19fe69ceab311eeb94b42e70e8c2070f9ba1025"
|
||||||
|
integrity sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==
|
||||||
|
dependencies:
|
||||||
|
forwarded "0.2.0"
|
||||||
|
ipaddr.js "1.9.1"
|
||||||
|
|
||||||
|
qs@6.13.0:
|
||||||
|
version "6.13.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/qs/-/qs-6.13.0.tgz#6ca3bd58439f7e245655798997787b0d88a51906"
|
||||||
|
integrity sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg==
|
||||||
|
dependencies:
|
||||||
|
side-channel "^1.0.6"
|
||||||
|
|
||||||
|
range-parser@~1.2.1:
|
||||||
|
version "1.2.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.1.tgz#3cf37023d199e1c24d1a55b84800c2f3e6468031"
|
||||||
|
integrity sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==
|
||||||
|
|
||||||
|
raw-body@2.5.2:
|
||||||
|
version "2.5.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.5.2.tgz#99febd83b90e08975087e8f1f9419a149366b68a"
|
||||||
|
integrity sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA==
|
||||||
|
dependencies:
|
||||||
|
bytes "3.1.2"
|
||||||
|
http-errors "2.0.0"
|
||||||
|
iconv-lite "0.4.24"
|
||||||
|
unpipe "1.0.0"
|
||||||
|
|
||||||
|
safe-buffer@5.2.1:
|
||||||
|
version "5.2.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6"
|
||||||
|
integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==
|
||||||
|
|
||||||
|
"safer-buffer@>= 2.1.2 < 3":
|
||||||
|
version "2.1.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a"
|
||||||
|
integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==
|
||||||
|
|
||||||
|
send@0.19.0:
|
||||||
|
version "0.19.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/send/-/send-0.19.0.tgz#bbc5a388c8ea6c048967049dbeac0e4a3f09d7f8"
|
||||||
|
integrity sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw==
|
||||||
|
dependencies:
|
||||||
|
debug "2.6.9"
|
||||||
|
depd "2.0.0"
|
||||||
|
destroy "1.2.0"
|
||||||
|
encodeurl "~1.0.2"
|
||||||
|
escape-html "~1.0.3"
|
||||||
|
etag "~1.8.1"
|
||||||
|
fresh "0.5.2"
|
||||||
|
http-errors "2.0.0"
|
||||||
|
mime "1.6.0"
|
||||||
|
ms "2.1.3"
|
||||||
|
on-finished "2.4.1"
|
||||||
|
range-parser "~1.2.1"
|
||||||
|
statuses "2.0.1"
|
||||||
|
|
||||||
|
serve-static@1.16.2:
|
||||||
|
version "1.16.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.16.2.tgz#b6a5343da47f6bdd2673848bf45754941e803296"
|
||||||
|
integrity sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw==
|
||||||
|
dependencies:
|
||||||
|
encodeurl "~2.0.0"
|
||||||
|
escape-html "~1.0.3"
|
||||||
|
parseurl "~1.3.3"
|
||||||
|
send "0.19.0"
|
||||||
|
|
||||||
|
set-function-length@^1.2.1:
|
||||||
|
version "1.2.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/set-function-length/-/set-function-length-1.2.2.tgz#aac72314198eaed975cf77b2c3b6b880695e5449"
|
||||||
|
integrity sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg==
|
||||||
|
dependencies:
|
||||||
|
define-data-property "^1.1.4"
|
||||||
|
es-errors "^1.3.0"
|
||||||
|
function-bind "^1.1.2"
|
||||||
|
get-intrinsic "^1.2.4"
|
||||||
|
gopd "^1.0.1"
|
||||||
|
has-property-descriptors "^1.0.2"
|
||||||
|
|
||||||
|
setprototypeof@1.2.0:
|
||||||
|
version "1.2.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.2.0.tgz#66c9a24a73f9fc28cbe66b09fed3d33dcaf1b424"
|
||||||
|
integrity sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==
|
||||||
|
|
||||||
|
side-channel@^1.0.6:
|
||||||
|
version "1.0.6"
|
||||||
|
resolved "https://registry.yarnpkg.com/side-channel/-/side-channel-1.0.6.tgz#abd25fb7cd24baf45466406b1096b7831c9215f2"
|
||||||
|
integrity sha512-fDW/EZ6Q9RiO8eFG8Hj+7u/oW+XrPTIChwCOM2+th2A6OblDtYYIpve9m+KvI9Z4C9qSEXlaGR6bTEYHReuglA==
|
||||||
|
dependencies:
|
||||||
|
call-bind "^1.0.7"
|
||||||
|
es-errors "^1.3.0"
|
||||||
|
get-intrinsic "^1.2.4"
|
||||||
|
object-inspect "^1.13.1"
|
||||||
|
|
||||||
|
statuses@2.0.1:
|
||||||
|
version "2.0.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/statuses/-/statuses-2.0.1.tgz#55cb000ccf1d48728bd23c685a063998cf1a1b63"
|
||||||
|
integrity sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==
|
||||||
|
|
||||||
|
toidentifier@1.0.1:
|
||||||
|
version "1.0.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/toidentifier/-/toidentifier-1.0.1.tgz#3be34321a88a820ed1bd80dfaa33e479fbb8dd35"
|
||||||
|
integrity sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==
|
||||||
|
|
||||||
|
type-is@~1.6.18:
|
||||||
|
version "1.6.18"
|
||||||
|
resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.18.tgz#4e552cd05df09467dcbc4ef739de89f2cf37c131"
|
||||||
|
integrity sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==
|
||||||
|
dependencies:
|
||||||
|
media-typer "0.3.0"
|
||||||
|
mime-types "~2.1.24"
|
||||||
|
|
||||||
|
unpipe@1.0.0, unpipe@~1.0.0:
|
||||||
|
version "1.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec"
|
||||||
|
integrity sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==
|
||||||
|
|
||||||
|
utils-merge@1.0.1:
|
||||||
|
version "1.0.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713"
|
||||||
|
integrity sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==
|
||||||
|
|
||||||
|
vary@^1, vary@~1.1.2:
|
||||||
|
version "1.1.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc"
|
||||||
|
integrity sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==
|
||||||
@@ -22,6 +22,65 @@
|
|||||||
|
|
||||||
<link rel="icon" href="favicon.ico" />
|
<link rel="icon" href="favicon.ico" />
|
||||||
|
|
||||||
|
<link rel="stylesheet" href="/fa/css/fontawesome.css" />
|
||||||
|
<link rel="stylesheet" href="/fa/css/brands.css" />
|
||||||
|
<link rel="stylesheet" href="/fa/css/regular.css" />
|
||||||
|
<link rel="stylesheet" href="/fa/css/solid.css" />
|
||||||
|
|
||||||
|
<!-- Preloads -->
|
||||||
|
<link rel="preload" href="fonts/Quicksand-Bold.woff2" as="font" type="font/woff2" crossorigin />
|
||||||
|
|
||||||
|
<link
|
||||||
|
rel="preload"
|
||||||
|
href="/fonts/Quicksand-Light.woff2"
|
||||||
|
as="font"
|
||||||
|
type="font/woff2"
|
||||||
|
crossorigin
|
||||||
|
/>
|
||||||
|
|
||||||
|
<link
|
||||||
|
rel="preload"
|
||||||
|
href="/fonts/Quicksand-Medium.woff2"
|
||||||
|
as="font"
|
||||||
|
type="font/woff2"
|
||||||
|
crossorigin
|
||||||
|
/>
|
||||||
|
|
||||||
|
<link
|
||||||
|
rel="preload"
|
||||||
|
href="/fonts/Quicksand-Regular.woff2"
|
||||||
|
as="font"
|
||||||
|
type="font/woff2"
|
||||||
|
crossorigin
|
||||||
|
/>
|
||||||
|
|
||||||
|
<link
|
||||||
|
rel="preload"
|
||||||
|
href="/fonts/Quicksand-SemiBold.woff2"
|
||||||
|
as="font"
|
||||||
|
type="font/woff2"
|
||||||
|
crossorigin
|
||||||
|
/>
|
||||||
|
|
||||||
|
<link rel="preload" as="image" href="/images/icon-pl.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/stacjownik-header-logo.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-dispatcher.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-train.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-arrow-asc.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-arrow-desc.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-filter2.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-stats.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-gnr.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-pojazdownik.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-diamond.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-user.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-like.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-spawn.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-timetableAll.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-timetableUnconfirmed.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-timetableConfirmed.svg" />
|
||||||
|
<link rel="preload" as="image" href="/images/icon-discord.png" />
|
||||||
|
|
||||||
<!-- Static OpenGraph meta -->
|
<!-- Static OpenGraph meta -->
|
||||||
<meta name="description" content="Pomocnik maszynisty i dyżurnego symulatora Train Driver 2" />
|
<meta name="description" content="Pomocnik maszynisty i dyżurnego symulatora Train Driver 2" />
|
||||||
<meta property="og:url" content="https://stacjownik-td2.web.app/" />
|
<meta property="og:url" content="https://stacjownik-td2.web.app/" />
|
||||||
@@ -31,10 +90,12 @@
|
|||||||
property="og:description"
|
property="og:description"
|
||||||
content="Pomocnik maszynisty i dyżurnego symulatora Train Driver 2"
|
content="Pomocnik maszynisty i dyżurnego symulatora Train Driver 2"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<meta
|
<meta
|
||||||
property="og:image"
|
property="og:image"
|
||||||
content="https://raw.githubusercontent.com/Spythere/api/main/thumbnails/stacjownik.jpg"
|
content="https://raw.githubusercontent.com/Spythere/api/main/thumbnails/stacjownik.jpg"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<meta property="og:image:width" content="1200" />
|
<meta property="og:image:width" content="1200" />
|
||||||
<meta property="og:image:height" content="630" />
|
<meta property="og:image:height" content="630" />
|
||||||
<meta property="og:site_name" content="Stacjownik" />
|
<meta property="og:site_name" content="Stacjownik" />
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "stacjownik",
|
"name": "stacjownik",
|
||||||
"version": "1.28.0",
|
"version": "1.30.7",
|
||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite --mode staging",
|
||||||
|
"dev:mock": "vite --mode development & yarn --cwd ./api-mock start",
|
||||||
|
"dev:fetch": "yarn --cwd ./api-mock fetch",
|
||||||
"build": "vue-tsc --noEmit && vite build",
|
"build": "vue-tsc --noEmit && vite build",
|
||||||
"deploy:prod": "yarn build && firebase deploy --only hosting",
|
"deploy:prod": "yarn build && firebase deploy --only hosting",
|
||||||
"deploy:dev": "yarn build && firebase hosting:channel:deploy dev --expires 7d",
|
"deploy:dev": "yarn build && firebase hosting:channel:deploy dev --expires 7d",
|
||||||
@@ -14,27 +16,27 @@
|
|||||||
"format": "prettier --write src/"
|
"format": "prettier --write src/"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"core-js": "^3.32.2",
|
"core-js": "^3.42.0",
|
||||||
"dotenv": "^16.3.1",
|
"dotenv": "^17.2.2",
|
||||||
"pinia": "^2.1.6",
|
"pinia": "^3.0.2",
|
||||||
"sass": "^1.67.0",
|
"sass": "^1.87.0",
|
||||||
"showdown": "^2.1.0",
|
"showdown": "^2.1.0",
|
||||||
"vue": "^3.3.4",
|
"vue": "^3.3.4",
|
||||||
"vue-i18n": "^9.4.1",
|
"vue-i18n": "^11.1.3",
|
||||||
"vue-router": "^4.4.0"
|
"vue-router": "^4.4.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/node": "^20.14.12",
|
"@types/node": "^24.3.1",
|
||||||
"@types/showdown": "^2.0.6",
|
"@types/showdown": "^2.0.6",
|
||||||
"@vite-pwa/assets-generator": "^0.2.4",
|
"@vite-pwa/assets-generator": "^1.0.0",
|
||||||
"@vitejs/plugin-vue": "^5.1.0",
|
"@vitejs/plugin-vue": "^6.0.1",
|
||||||
"@vue/tsconfig": "^0.5.1",
|
"@vue/tsconfig": "^0.8.1",
|
||||||
"axios": "^1.7.2",
|
"axios": "^1.9.0",
|
||||||
"prettier": "^3.3.3",
|
"prettier": "^3.3.3",
|
||||||
"typescript": "^5.5.4",
|
"typescript": "^5.5.4",
|
||||||
"vite": "^5.3.4",
|
"vite": "^7.1.4",
|
||||||
"vite-plugin-pwa": "^0.20.0",
|
"vite-plugin-pwa": "^1.0.0",
|
||||||
"vue-tsc": "^2.0.28"
|
"vue-tsc": "^3.0.6"
|
||||||
},
|
},
|
||||||
"browserslist": [
|
"browserslist": [
|
||||||
"> 1%",
|
"> 1%",
|
||||||
|
|||||||
@@ -0,0 +1,19 @@
|
|||||||
|
/*!
|
||||||
|
* Font Awesome Free 6.7.2 by @fontawesome - https://fontawesome.com
|
||||||
|
* License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License)
|
||||||
|
* Copyright 2024 Fonticons, Inc.
|
||||||
|
*/
|
||||||
|
:root, :host {
|
||||||
|
--fa-style-family-classic: 'Font Awesome 6 Free';
|
||||||
|
--fa-font-regular: normal 400 1em/1 'Font Awesome 6 Free'; }
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Font Awesome 6 Free';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 400;
|
||||||
|
font-display: block;
|
||||||
|
src: url("../webfonts/fa-regular-400.woff2") format("woff2"), url("../webfonts/fa-regular-400.ttf") format("truetype"); }
|
||||||
|
|
||||||
|
.far,
|
||||||
|
.fa-regular {
|
||||||
|
font-weight: 400; }
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
/*!
|
||||||
|
* Font Awesome Free 6.7.2 by @fontawesome - https://fontawesome.com
|
||||||
|
* License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License)
|
||||||
|
* Copyright 2024 Fonticons, Inc.
|
||||||
|
*/
|
||||||
|
:root, :host {
|
||||||
|
--fa-style-family-classic: 'Font Awesome 6 Free';
|
||||||
|
--fa-font-solid: normal 900 1em/1 'Font Awesome 6 Free'; }
|
||||||
|
|
||||||
|
@font-face {
|
||||||
|
font-family: 'Font Awesome 6 Free';
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 900;
|
||||||
|
font-display: block;
|
||||||
|
src: url("../webfonts/fa-solid-900.woff2") format("woff2"), url("../webfonts/fa-solid-900.ttf") format("truetype"); }
|
||||||
|
|
||||||
|
.fas,
|
||||||
|
.fa-solid {
|
||||||
|
font-weight: 900; }
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
<svg
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width="2"
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
>
|
||||||
|
<path d="m21.73 18-8-14a2 2 0 0 0-3.48 0l-8 14A2 2 0 0 0 4 21h16a2 2 0 0 0 1.73-3" />
|
||||||
|
<path d="M12 9v4" />
|
||||||
|
<path d="M12 17h.01" />
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 345 B |
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="144" height="144" viewBox="0 0 144 144" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M82.7143 61.3284L118.429 7L22 74.9104H68.4286L36.2857 137L122 61.3284H82.7143Z" fill="#FFF500"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 213 B |
@@ -0,0 +1,7 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" id="flag-icons-gb" viewBox="0 0 640 480">
|
||||||
|
<path fill="#012169" d="M0 0h640v480H0z"/>
|
||||||
|
<path fill="#FFF" d="m75 0 244 181L562 0h78v62L400 241l240 178v61h-80L320 301 81 480H0v-60l239-178L0 64V0z"/>
|
||||||
|
<path fill="#C8102E" d="m424 281 216 159v40L369 281zm-184 20 6 35L54 480H0zM640 0v3L391 191l2-44L590 0zM0 0l239 176h-60L0 42z"/>
|
||||||
|
<path fill="#FFF" d="M241 0v480h160V0zM0 160v160h640V160z"/>
|
||||||
|
<path fill="#C8102E" d="M0 193v96h640v-96zM273 0v480h96V0z"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 504 B |
@@ -0,0 +1,47 @@
|
|||||||
|
<svg width="512" height="512" viewBox="0 0 512 512" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<rect width="512" height="512" rx="256" fill="black"/>
|
||||||
|
<rect x="329.454" y="340.498" width="7.7497" height="147.239" rx="3.87485" transform="rotate(90 329.454 340.498)" fill="white"/>
|
||||||
|
<rect x="308" y="320" width="5" height="103" rx="2.5" transform="rotate(90 308 320)" fill="white"/>
|
||||||
|
<rect x="366.263" y="367.622" width="11.6246" height="213.496" rx="5.81228" transform="rotate(90 366.263 367.622)" fill="white"/>
|
||||||
|
<g filter="url(#filter0_d_1067_42)">
|
||||||
|
<rect width="18.2931" height="124.137" rx="9.14654" transform="matrix(0.688736 0.725012 -0.688736 0.725012 212.498 294)" fill="white"/>
|
||||||
|
</g>
|
||||||
|
<g filter="url(#filter1_d_1067_42)">
|
||||||
|
<rect width="19.6916" height="124.137" rx="9.84578" transform="matrix(-0.688736 0.725012 0.688736 0.725012 303.502 294)" fill="white"/>
|
||||||
|
</g>
|
||||||
|
<g filter="url(#filter2_d_1067_42)">
|
||||||
|
<path d="M147.893 304.935H121.015C109.69 304.935 101.233 302.498 95.6422 297.624C90.195 292.607 87.4713 285.01 87.4713 274.832V194.628C87.4713 184.307 90.195 176.71 95.6422 171.836C101.233 166.962 109.69 164.525 121.015 164.525H178.856V187.318H115.854C114.134 187.318 113.274 188.178 113.274 189.898V279.562C113.274 281.283 114.134 282.143 115.854 282.143H154.559C156.279 282.143 157.139 281.283 157.139 279.562V245.589L159.719 249.674H138.002V228.387H181.436V274.832C181.436 285.01 178.641 292.607 173.051 297.624C167.603 302.498 159.217 304.935 147.893 304.935ZM282.921 265.371V166.46H304.853V303H287.006L222.284 204.734L226.585 203.874V303H204.867V166.46H222.499L287.006 264.511L282.921 265.371ZM358.59 303H333.218V166.46L391.059 165.6C402.527 165.313 411.199 167.894 417.077 173.341C423.097 178.788 426.108 186.672 426.108 196.994V218.711C426.108 227.025 423.814 233.978 419.227 239.568C414.783 245.159 408.189 248.671 399.445 250.104V245.159C403.889 246.019 407.401 247.739 409.981 250.319C412.561 252.9 414.783 256.842 416.647 262.146L431.053 303H405.68L389.339 254.19C388.909 253.043 388.479 252.255 388.049 251.824C387.762 251.251 386.973 250.964 385.683 250.964H354.505L358.59 246.879V303ZM358.59 183.232V233.117L354.935 229.247H397.295C399.301 229.247 400.305 228.315 400.305 226.452V190.328C400.305 188.464 399.301 187.533 397.295 187.533H354.935L358.59 183.232Z" fill="white"/>
|
||||||
|
</g>
|
||||||
|
<defs>
|
||||||
|
<filter id="filter0_d_1067_42" x="101.768" y="268.962" width="148.561" height="153.339" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
|
||||||
|
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
|
||||||
|
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
|
||||||
|
<feOffset/>
|
||||||
|
<feGaussianBlur stdDeviation="14.4611"/>
|
||||||
|
<feComposite in2="hardAlpha" operator="out"/>
|
||||||
|
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.25 0"/>
|
||||||
|
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1067_42"/>
|
||||||
|
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1067_42" result="shape"/>
|
||||||
|
</filter>
|
||||||
|
<filter id="filter1_d_1067_42" x="264.99" y="269.259" width="148.96" height="153.759" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
|
||||||
|
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
|
||||||
|
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
|
||||||
|
<feOffset/>
|
||||||
|
<feGaussianBlur stdDeviation="14.4611"/>
|
||||||
|
<feComposite in2="hardAlpha" operator="out"/>
|
||||||
|
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.25 0"/>
|
||||||
|
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1067_42"/>
|
||||||
|
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1067_42" result="shape"/>
|
||||||
|
</filter>
|
||||||
|
<filter id="filter2_d_1067_42" x="85.2844" y="161.245" width="352.33" height="149.158" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
|
||||||
|
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
|
||||||
|
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
|
||||||
|
<feOffset dx="2.18692" dy="1.09346"/>
|
||||||
|
<feGaussianBlur stdDeviation="2.18692"/>
|
||||||
|
<feComposite in2="hardAlpha" operator="out"/>
|
||||||
|
<feColorMatrix type="matrix" values="0 0 0 0 0.5375 0 0 0 0 0.5375 0 0 0 0 0.5375 0 0 0 1 0"/>
|
||||||
|
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1067_42"/>
|
||||||
|
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1067_42" result="shape"/>
|
||||||
|
</filter>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 4.3 KiB |
@@ -0,0 +1,60 @@
|
|||||||
|
<svg width="79" height="127" viewBox="0 0 79 127" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<g id="icon-loading">
|
||||||
|
<g id="Rectangle 37">
|
||||||
|
<rect id="Rectangle 38" x="36.9999" y="79" width="6" height="16" fill="#FF0000"/>
|
||||||
|
<rect id="Rectangle 40" x="36.9999" y="111" width="6" height="16" fill="#FF0000"/>
|
||||||
|
<rect id="Rectangle 39" x="36.9999" y="95" width="6" height="16" fill="white"/>
|
||||||
|
</g>
|
||||||
|
<g id="Group 8">
|
||||||
|
<path id="Vector 15" d="M59.5018 41.0003H23.0018C-1.49817 41.0003 -0.498171 79.5003 23.0018 79.5003H59.5018C83.0018 79.5003 84.0018 41.0003 59.5018 41.0003Z" fill="#3F3E3E"/>
|
||||||
|
<g id="Group 51">
|
||||||
|
<circle id="light-left" cx="22.9999" cy="60" r="9" fill="#FF1313" fill-opacity="1"/>
|
||||||
|
|
||||||
|
<animate
|
||||||
|
attributeType="XML"
|
||||||
|
attributeName="opacity"
|
||||||
|
values="0.25;1;1;0.25;0.25"
|
||||||
|
dur="1s"
|
||||||
|
repeatCount="indefinite"
|
||||||
|
/>
|
||||||
|
</g>
|
||||||
|
|
||||||
|
<g id="Group 51">
|
||||||
|
<circle id="light-right" cx="57.9999" cy="60" r="9" fill="#FF1313" fill-opacity="1"/>
|
||||||
|
|
||||||
|
<animate
|
||||||
|
attributeType="XML"
|
||||||
|
attributeName="opacity"
|
||||||
|
values="1;0.25;0.25;1;1"
|
||||||
|
dur="1s"
|
||||||
|
repeatCount="indefinite"
|
||||||
|
/>
|
||||||
|
</g>
|
||||||
|
|
||||||
|
|
||||||
|
</g>
|
||||||
|
<g id="Group 52">
|
||||||
|
<rect id="Rectangle 36" x="37.9999" y="10" width="4" height="31" fill="#525252"/>
|
||||||
|
<g id="Vector 23">
|
||||||
|
<path id="Rectangle 28" d="M4.09756 32.3241L10.9579 29.2933L14.1908 36.611L7.33047 39.6418L3.42724 36.9932L4.09756 32.3241Z" fill="#FF0000"/>
|
||||||
|
<path id="Rectangle 30" d="M10.9579 29.2933L20.105 25.2522L23.3379 32.5698L14.1908 36.611L12.5743 32.9521L10.9579 29.2933Z" fill="white"/>
|
||||||
|
<path id="Rectangle 34" d="M20.105 25.2522L29.2521 21.211L32.485 28.5287L23.3379 32.5698L21.7214 28.911L20.105 25.2522Z" fill="#FF0000"/>
|
||||||
|
<path id="Rectangle 35" d="M47.5463 13.1288L56.6934 9.08762L59.9263 16.4053L50.7792 20.4464L49.1627 16.7876L47.5463 13.1288Z" fill="#FF0000"/>
|
||||||
|
<path id="Rectangle 31" d="M29.2521 21.211L38.3992 17.1699L41.6321 24.4876L32.485 28.5287L30.8685 24.8699L29.2521 21.211Z" fill="white"/>
|
||||||
|
<path id="Rectangle 32" d="M38.3992 17.1699L47.5463 13.1288L50.7792 20.4464L41.6321 24.4876L40.0156 20.8287L38.3992 17.1699Z" fill="white"/>
|
||||||
|
<path id="Rectangle 33" d="M56.6934 9.08762L65.8404 5.04649L69.0734 12.3642L59.9263 16.4053L58.3098 12.7465L56.6934 9.08762Z" fill="white"/>
|
||||||
|
<path id="Rectangle 29" d="M73.1581 1.81359L65.8405 5.04649L69.0734 12.3642L76.391 9.13126L76.604 4.6642L73.1581 1.81359Z" fill="#FF0000"/>
|
||||||
|
</g>
|
||||||
|
<g id="Vector 24">
|
||||||
|
<path id="Rectangle 28_2" d="M6.36567 3.47438L13.3598 6.18222L10.4714 13.6426L3.47731 10.9348L2.59012 6.30195L6.36567 3.47438Z" fill="#FF0000"/>
|
||||||
|
<path id="Rectangle 30_2" d="M13.3597 6.18222L22.6852 9.79268L19.7969 17.2531L10.4714 13.6426L11.9156 9.91241L13.3597 6.18222Z" fill="white"/>
|
||||||
|
<path id="Rectangle 34_2" d="M22.6853 9.79268L32.0108 13.4031L29.1224 20.8635L19.7969 17.2531L21.2411 13.5229L22.6853 9.79268Z" fill="#FF0000"/>
|
||||||
|
<path id="Rectangle 35_2" d="M50.6617 20.6241L59.9872 24.2345L57.0989 31.6949L47.7734 28.0844L49.2176 24.3542L50.6617 20.6241Z" fill="#FF0000"/>
|
||||||
|
<path id="Rectangle 31_2" d="M32.0107 13.4031L41.3362 17.0136L38.4479 24.474L29.1224 20.8635L30.5666 17.1333L32.0107 13.4031Z" fill="white"/>
|
||||||
|
<path id="Rectangle 32_2" d="M41.3363 17.0136L50.6618 20.6241L47.7734 28.0844L38.4479 24.474L39.8921 20.7438L41.3363 17.0136Z" fill="white"/>
|
||||||
|
<path id="Rectangle 33_2" d="M59.9872 24.2345L69.3127 27.845L66.4243 35.3054L57.0988 31.6949L58.543 27.9647L59.9872 24.2345Z" fill="white"/>
|
||||||
|
<path id="Rectangle 29_2" d="M76.7731 30.7333L69.3127 27.845L66.4243 35.3054L73.8847 38.1937L77.194 35.1856L76.7731 30.7333Z" fill="#FF0000"/>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 3.4 KiB |
@@ -1,4 +1,6 @@
|
|||||||
<svg width="39" height="23" viewBox="0 0 39 23" fill="none" xmlns="http://www.w3.org/2000/svg">
|
<svg xmlns="http://www.w3.org/2000/svg" id="flag-icons-pl" viewBox="0 0 640 480">
|
||||||
<rect width="39" height="23" fill="#FF0F0F"/>
|
<g fill-rule="evenodd">
|
||||||
<rect width="39" height="11.5" fill="white"/>
|
<path fill="#fff" d="M640 480H0V0h640z"/>
|
||||||
|
<path fill="#dc143c" d="M640 480H0V240h640z"/>
|
||||||
|
</g>
|
||||||
</svg>
|
</svg>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 199 B After Width: | Height: | Size: 219 B |
@@ -0,0 +1,24 @@
|
|||||||
|
<svg width="500" height="500" viewBox="0 0 500 500" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<mask id="path-1-inside-1_102_63" fill="white">
|
||||||
|
<path d="M0 250C0 111.929 111.929 6.10352e-05 250 6.10352e-05C388.071 6.10352e-05 500 111.929 500 250C500 388.071 388.071 500 250 500C111.929 500 0 388.071 0 250Z"/>
|
||||||
|
</mask>
|
||||||
|
<path d="M0 250C0 111.929 111.929 6.10352e-05 250 6.10352e-05C388.071 6.10352e-05 500 111.929 500 250C500 388.071 388.071 500 250 500C111.929 500 0 388.071 0 250Z" fill="#242424"/>
|
||||||
|
<path d="M0 222.821C0 84.7503 111.929 -27.1785 250 -27.1785C388.071 -27.1785 500 84.7503 500 222.821V250C500 126.939 388.071 27.1787 250 27.1787C111.929 27.1787 0 126.939 0 250V222.821ZM500 277.179C500 415.25 388.071 527.179 250 527.179C111.929 527.179 0 415.25 0 277.179V250C0 373.061 111.929 472.821 250 472.821C388.071 472.821 500 373.061 500 250V277.179ZM0 500V6.10352e-05V500ZM500 6.10352e-05V500V6.10352e-05Z" fill="#FFD600" mask="url(#path-1-inside-1_102_63)"/>
|
||||||
|
<path d="M210.369 301.604C210.369 301.604 210.369 341.807 210.369 364.846C210.369 387.885 202.798 417.491 171.591 417.491C140.385 417.491 132.813 417.491 132.813 417.491L132.812 78.125L250.754 78.125C274.312 78.125 294.504 80.9665 311.331 86.6494C328.311 92.1788 342.232 99.8585 353.093 109.689C364.107 119.519 372.214 131.115 377.415 144.478C382.616 157.84 385.217 172.278 385.217 187.791C385.217 204.533 382.54 219.892 377.186 233.869C371.832 247.846 363.648 259.827 352.634 269.81C341.62 279.794 327.623 287.627 310.643 293.31C293.816 298.839 273.853 301.604 250.754 301.604L210.369 301.604ZM210.369 242.854L250.754 242.854C270.946 242.854 285.479 238.016 294.351 228.34C303.224 218.663 307.66 205.147 307.66 187.791C307.66 180.111 306.512 173.123 304.218 166.825C301.923 160.528 298.405 155.152 293.663 150.698C289.074 146.09 283.184 142.558 275.995 140.1C268.958 137.643 260.544 136.414 250.754 136.414L210.369 136.414L210.369 242.854Z" fill="url(#paint0_linear_102_63)"/>
|
||||||
|
<path d="M239.215 301.604C239.215 301.604 239.215 341.807 239.215 364.846C239.215 387.885 231.643 417.491 200.437 417.491C169.231 417.491 161.659 417.491 161.659 417.491L161.658 78.125L279.6 78.125C303.158 78.125 323.35 80.9665 340.177 86.6494C357.157 92.1788 371.077 99.8585 381.938 109.689C392.952 119.519 401.06 131.115 406.261 144.478C411.462 157.84 414.062 172.278 414.062 187.791C414.062 204.533 411.385 219.892 406.031 233.869C400.677 247.846 392.493 259.827 381.479 269.81C370.465 279.794 356.468 287.627 339.488 293.31C322.662 298.839 302.699 301.604 279.6 301.604L239.215 301.604ZM239.215 242.854L279.6 242.854C299.792 242.854 314.325 238.016 323.197 228.34C332.069 218.663 336.505 205.147 336.505 187.791C336.505 180.111 335.358 173.123 333.064 166.825C330.769 160.528 327.251 155.152 322.509 150.698C317.919 146.09 312.03 142.558 304.84 140.1C297.804 137.643 289.39 136.414 279.6 136.414L239.215 136.414L239.215 242.854Z" fill="url(#paint1_linear_102_63)"/>
|
||||||
|
<path d="M210.685 301.604C210.685 301.604 210.685 341.807 210.685 364.846C210.685 387.885 203.082 417.491 171.749 417.491C140.416 417.491 132.813 417.491 132.813 417.491L132.812 78.125L251.233 78.125C274.887 78.125 295.161 80.9665 312.057 86.6494C329.105 92.1788 343.083 99.8585 353.988 109.689C365.046 119.519 373.187 131.115 378.409 144.478C383.631 157.84 386.242 172.278 386.242 187.791C386.242 204.533 383.555 219.892 378.179 233.869C372.803 247.846 364.586 259.827 353.527 269.81C342.468 279.794 328.414 287.627 311.365 293.31C294.47 298.839 274.426 301.604 251.233 301.604L210.685 301.604ZM210.685 242.854L251.233 242.854C271.508 242.854 286.099 238.016 295.008 228.34C303.916 218.663 308.37 205.147 308.37 187.791C308.37 180.111 307.218 173.123 304.914 166.825C302.611 160.528 299.078 155.152 294.316 150.698C289.709 146.09 283.795 142.558 276.576 140.1C269.511 137.643 261.063 136.414 251.233 136.414L210.685 136.414L210.685 242.854Z" fill="url(#paint2_radial_102_63)"/>
|
||||||
|
<defs>
|
||||||
|
<linearGradient id="paint0_linear_102_63" x1="259.015" y1="78.125" x2="259.015" y2="417.491" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop offset="0.135417" stop-color="#FFD600"/>
|
||||||
|
<stop offset="1"/>
|
||||||
|
</linearGradient>
|
||||||
|
<linearGradient id="paint1_linear_102_63" x1="287.86" y1="78.125" x2="287.86" y2="417.491" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop offset="0.135417" stop-color="#FFD600"/>
|
||||||
|
<stop offset="1"/>
|
||||||
|
</linearGradient>
|
||||||
|
<radialGradient id="paint2_radial_102_63" cx="0" cy="0" r="1" gradientUnits="userSpaceOnUse" gradientTransform="translate(259.527 247.808) rotate(0.36307) scale(345.948 325.206)">
|
||||||
|
<stop offset="0.484375" stop-color="white"/>
|
||||||
|
<stop offset="1"/>
|
||||||
|
</radialGradient>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 4.5 KiB |
@@ -0,0 +1,5 @@
|
|||||||
|
<svg width="144" height="144" viewBox="0 0 144 144" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<rect x="9.20437" y="2.4661" width="36.5457" height="57.8287" rx="16.7449" transform="matrix(0.869001 -0.494811 0.505207 0.862998 50.006 87.4256)" stroke="white" stroke-width="13.3959"/>
|
||||||
|
<rect x="9.20437" y="2.4661" width="36.5457" height="57.8287" rx="16.7449" transform="matrix(0.869001 -0.494811 0.505207 0.862998 14.9599 29.6039)" stroke="white" stroke-width="13.3959"/>
|
||||||
|
<path d="M65.1133 58.145L79.8524 84.3103" stroke="white" stroke-width="10.0469" stroke-linecap="round" stroke-linejoin="bevel"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 611 B |
@@ -0,0 +1,5 @@
|
|||||||
|
<svg width="512" height="512" viewBox="0 0 512 512" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<rect width="512" height="512" rx="256" fill="#151414"/>
|
||||||
|
<path d="M72.4253 291.986V279.965H120.201C123.283 279.965 124.824 278.424 124.824 275.342V264.246C124.824 261.266 123.54 259.571 120.971 259.16L90.9189 252.995C78.5898 250.529 72.4253 242.259 72.4253 228.183V219.553C72.4253 202.292 81.0557 193.662 98.3164 193.662H133.608L143.934 201.675V213.696H99.2411C96.1588 213.696 94.6177 215.237 94.6177 218.32V228.337C94.6177 231.214 95.9019 232.909 98.4705 233.423L128.523 239.433C140.852 241.899 147.016 250.17 147.016 264.246V274.109C147.016 291.37 138.386 300 121.125 300H82.7509L72.4253 291.986ZM167.651 300V193.662H219.433C236.694 193.662 245.324 202.292 245.324 219.553V237.122C245.324 249.964 240.546 257.978 230.991 261.163L248.406 295.377L245.786 300H226.676L207.874 263.013H189.843V300H167.651ZM189.843 242.978H218.508C221.591 242.978 223.132 241.437 223.132 238.355V218.32C223.132 215.237 221.591 213.696 218.508 213.696H189.843V242.978ZM262.96 274.109V253.766H285.153V275.342C285.153 278.424 286.694 279.965 289.776 279.965H310.736C313.818 279.965 315.359 278.424 315.359 275.342V213.696H286.386V193.662H337.551V274.109C337.551 291.37 328.921 300 311.66 300H288.852C271.591 300 262.96 291.37 262.96 274.109ZM361.948 300V193.662H413.731C430.991 193.662 439.622 202.292 439.622 219.553V240.204C439.622 257.465 430.991 266.095 413.731 266.095H384.141V300H361.948ZM384.141 246.06H412.806C415.888 246.06 417.429 244.519 417.429 241.437V218.32C417.429 215.237 415.888 213.696 412.806 213.696H384.141V246.06Z" fill="white"/>
|
||||||
|
<path d="M304.958 332.848V322.831H348.418V332.848H332.236V376H321.14V332.848H304.958ZM356.61 376V322.831H376.799C391.285 322.831 398.529 330.074 398.529 344.561V354.27C398.529 368.757 391.285 376 376.799 376H356.61ZM367.706 365.983H377.415C384.093 365.983 387.432 362.643 387.432 355.965V342.866C387.432 336.187 384.093 332.848 377.415 332.848H367.706V365.983ZM407.35 376V358.662C407.35 351.624 410.432 347.489 416.597 346.256L430.852 343.405C432.136 343.148 432.779 342.3 432.779 340.862V335.16C432.779 333.619 432.008 332.848 430.467 332.848H408.891V326.838L414.054 322.831H430.929C439.56 322.831 443.875 327.146 443.875 335.776V340.785C443.875 347.823 440.792 351.958 434.628 353.191L420.372 356.042C419.088 356.299 418.446 357.147 418.446 358.585V365.983H443.875V376H407.35Z" fill="#E63E3E"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 2.4 KiB |
|
After Width: | Height: | Size: 30 KiB |
|
After Width: | Height: | Size: 18 KiB |
@@ -1,203 +1,21 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="app_container">
|
<div class="app_container">
|
||||||
<UpdateCard
|
<AppNewDomainInfo />
|
||||||
:is-update-card-open="isUpdateCardOpen"
|
|
||||||
@toggle-card="() => (isUpdateCardOpen = false)"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<Tooltip />
|
|
||||||
<AppHeader :current-lang="currentLang" @change-lang="changeLang" />
|
|
||||||
|
|
||||||
<main class="app_main">
|
|
||||||
<router-view v-slot="{ Component }">
|
|
||||||
<keep-alive exclude="SceneryView">
|
|
||||||
<component :is="Component" :key="$route.name" />
|
|
||||||
</keep-alive>
|
|
||||||
</router-view>
|
|
||||||
</main>
|
|
||||||
|
|
||||||
<footer class="app_footer">
|
|
||||||
©
|
|
||||||
<a href="https://td2.info.pl/profile/?u=20777" target="_blank">Spythere</a>
|
|
||||||
{{ new Date().getUTCFullYear() }} |
|
|
||||||
<button class="btn--text" @click="() => (isUpdateCardOpen = true)">
|
|
||||||
v{{ VERSION }}{{ isOnProductionHost ? '' : 'dev' }}
|
|
||||||
</button>
|
|
||||||
|
|
||||||
<br />
|
|
||||||
<a href="https://discord.gg/x2mpNN3svk">
|
|
||||||
<img src="/images/icon-discord.png" alt="" /> <b>{{ $t('footer.discord') }}</b>
|
|
||||||
</a>
|
|
||||||
|
|
||||||
<div style="display: none">∫ ukryta taktyczna całka do programowania w HTMLu</div>
|
|
||||||
</footer>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts" setup>
|
||||||
import { defineComponent } from 'vue';
|
import AppNewDomainInfo from './components/App/AppNewDomainInfo.vue';
|
||||||
import axios from 'axios';
|
|
||||||
|
|
||||||
import { version } from '.././package.json';
|
|
||||||
import { Status } from './typings/common';
|
|
||||||
import { useMainStore } from './store/mainStore';
|
|
||||||
import { useApiStore } from './store/apiStore';
|
|
||||||
import { useTooltipStore } from './store/tooltipStore';
|
|
||||||
|
|
||||||
import Clock from './components/App/Clock.vue';
|
|
||||||
import StatusIndicator from './components/App/StatusIndicator.vue';
|
|
||||||
import AppHeader from './components/App/AppHeader.vue';
|
|
||||||
import Tooltip from './components/Tooltip/Tooltip.vue';
|
|
||||||
import UpdateCard from './components/App/UpdateCard.vue';
|
|
||||||
|
|
||||||
import StorageManager from './managers/storageManager';
|
|
||||||
|
|
||||||
const STORAGE_VERSION_KEY = 'app_version';
|
|
||||||
|
|
||||||
export default defineComponent({
|
|
||||||
components: {
|
|
||||||
Clock,
|
|
||||||
StatusIndicator,
|
|
||||||
AppHeader,
|
|
||||||
UpdateCard,
|
|
||||||
Tooltip
|
|
||||||
},
|
|
||||||
|
|
||||||
data: () => ({
|
|
||||||
VERSION: version,
|
|
||||||
store: useMainStore(),
|
|
||||||
apiStore: useApiStore(),
|
|
||||||
tooltipStore: useTooltipStore(),
|
|
||||||
|
|
||||||
isUpdateCardOpen: false,
|
|
||||||
|
|
||||||
currentLang: 'pl',
|
|
||||||
isOnProductionHost: location.hostname == 'stacjownik-td2.web.app'
|
|
||||||
}),
|
|
||||||
|
|
||||||
created() {
|
|
||||||
this.init();
|
|
||||||
},
|
|
||||||
|
|
||||||
async mounted() {
|
|
||||||
window.addEventListener('mousemove', (e: MouseEvent) => this.tooltipStore.handle(e));
|
|
||||||
},
|
|
||||||
|
|
||||||
methods: {
|
|
||||||
init() {
|
|
||||||
if (!this.isOnProductionHost) document.title = 'Stacjownik Dev';
|
|
||||||
|
|
||||||
this.loadLang();
|
|
||||||
this.setupOfflineHandling();
|
|
||||||
this.checkAppVersion();
|
|
||||||
|
|
||||||
this.apiStore.setupAPIData();
|
|
||||||
},
|
|
||||||
|
|
||||||
async checkAppVersion() {
|
|
||||||
const storageVersion = StorageManager.getStringValue(STORAGE_VERSION_KEY);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const releaseData = await (
|
|
||||||
await axios.get('https://api.github.com/repos/Spythere/stacjownik/releases/latest')
|
|
||||||
).data;
|
|
||||||
|
|
||||||
if (!releaseData) return;
|
|
||||||
|
|
||||||
this.store.appUpdate = {
|
|
||||||
version,
|
|
||||||
changelog: releaseData.body,
|
|
||||||
releaseURL: releaseData.html_url
|
|
||||||
};
|
|
||||||
|
|
||||||
this.isUpdateCardOpen =
|
|
||||||
(storageVersion != '' && storageVersion != version && this.isOnProductionHost) ||
|
|
||||||
import.meta.env.VITE_UPDATE_TEST === 'test';
|
|
||||||
} catch (error) {
|
|
||||||
console.error(`Wystąpił błąd podczas pobierania danych z API GitHuba: ${error}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
StorageManager.setStringValue(STORAGE_VERSION_KEY, version);
|
|
||||||
},
|
|
||||||
|
|
||||||
setupOfflineHandling() {
|
|
||||||
this.store.isOffline = !window.navigator.onLine;
|
|
||||||
|
|
||||||
if (this.store.isOffline) this.handleOfflineMode();
|
|
||||||
|
|
||||||
window.addEventListener('offline', this.handleOfflineMode);
|
|
||||||
window.addEventListener('online', this.handleOnlineMode);
|
|
||||||
},
|
|
||||||
|
|
||||||
handleOfflineMode() {
|
|
||||||
this.store.isOffline = true;
|
|
||||||
|
|
||||||
this.apiStore.activeData = undefined;
|
|
||||||
this.apiStore.dataStatuses.connection = Status.Data.Offline;
|
|
||||||
},
|
|
||||||
|
|
||||||
handleOnlineMode() {
|
|
||||||
this.store.isOffline = false;
|
|
||||||
this.apiStore.dataStatuses.connection = Status.Data.Loading;
|
|
||||||
|
|
||||||
this.apiStore.connectToAPI();
|
|
||||||
},
|
|
||||||
|
|
||||||
changeLang(lang: string) {
|
|
||||||
this.$i18n.locale = lang;
|
|
||||||
this.currentLang = lang;
|
|
||||||
|
|
||||||
StorageManager.setStringValue('lang', lang);
|
|
||||||
},
|
|
||||||
|
|
||||||
loadLang() {
|
|
||||||
const storageLang = StorageManager.getStringValue('lang');
|
|
||||||
|
|
||||||
if (storageLang) {
|
|
||||||
this.changeLang(storageLang);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!window.navigator.language) return;
|
|
||||||
|
|
||||||
const naviLanguage = window.navigator.language.toString();
|
|
||||||
|
|
||||||
if (naviLanguage.startsWith('en')) {
|
|
||||||
this.changeLang('en');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
@import './styles/global';
|
@use './styles/animations';
|
||||||
@import './styles/animations';
|
|
||||||
|
|
||||||
.route {
|
|
||||||
margin: 0 0.2em;
|
|
||||||
|
|
||||||
&-active,
|
|
||||||
&[data-active='true'] {
|
|
||||||
color: $accentCol;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// APP
|
// APP
|
||||||
#app {
|
#app {
|
||||||
color: white;
|
color: white;
|
||||||
font-size: 1rem;
|
|
||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
|
font-size: 1em;
|
||||||
@include smallScreen() {
|
|
||||||
font-size: calc(0.65rem + 0.85vw);
|
|
||||||
}
|
|
||||||
|
|
||||||
@include screenLandscape() {
|
|
||||||
font-size: calc(0.45rem + 0.8vw);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// CONTAINER
|
// CONTAINER
|
||||||
@@ -214,38 +32,4 @@ export default defineComponent({
|
|||||||
.app_main {
|
.app_main {
|
||||||
padding: 0 0.5em;
|
padding: 0 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.warning {
|
|
||||||
background-color: firebrick;
|
|
||||||
text-align: center;
|
|
||||||
padding: 0.5em 0.4em;
|
|
||||||
max-width: 1100px;
|
|
||||||
margin: 0 auto;
|
|
||||||
|
|
||||||
border-radius: 0 0 1em 1em;
|
|
||||||
}
|
|
||||||
|
|
||||||
// FOOTER
|
|
||||||
.app_footer {
|
|
||||||
max-width: 100%;
|
|
||||||
padding: 0.5em;
|
|
||||||
|
|
||||||
button {
|
|
||||||
display: inline-block;
|
|
||||||
padding: 0.1em;
|
|
||||||
}
|
|
||||||
|
|
||||||
img {
|
|
||||||
width: 1.1em;
|
|
||||||
vertical-align: text-bottom;
|
|
||||||
}
|
|
||||||
|
|
||||||
z-index: 10;
|
|
||||||
|
|
||||||
background: #111;
|
|
||||||
color: white;
|
|
||||||
|
|
||||||
text-align: center;
|
|
||||||
vertical-align: middle;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -0,0 +1,41 @@
|
|||||||
|
<template>
|
||||||
|
<footer class="app_footer">
|
||||||
|
©
|
||||||
|
<a href="https://td2.info.pl/profile/?u=20777" target="_blank">Spythere</a>
|
||||||
|
{{ new Date().getUTCFullYear() }} |
|
||||||
|
<button class="btn--text" @click="openUpdateCard">
|
||||||
|
v{{ version }}{{ isOnProductionHost ? '' : 'dev' }}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<br />
|
||||||
|
<a href="https://discord.gg/x2mpNN3svk">
|
||||||
|
<img src="/images/icon-discord.png" alt="discord logo icon" /> <b class="text--discord">
|
||||||
|
{{ $t('footer.discord') }}
|
||||||
|
</b>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<div style="display: none">∫ ukryta taktyczna całka do programowania w HTMLu</div>
|
||||||
|
</footer>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent } from 'vue';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
emits: ['openUpdateCard'],
|
||||||
|
props: {
|
||||||
|
isUpdateCardOpen: {
|
||||||
|
type: Boolean,
|
||||||
|
required: true
|
||||||
|
},
|
||||||
|
version: String,
|
||||||
|
isOnProductionHost: Boolean
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
openUpdateCard() {
|
||||||
|
this.$emit('openUpdateCard');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
@@ -1,24 +1,17 @@
|
|||||||
<template>
|
<template>
|
||||||
<header class="app_header">
|
<header class="app_header">
|
||||||
<div class="header_container">
|
<div class="header_container">
|
||||||
<div class="header_icons">
|
|
||||||
<span class="icons-top">
|
|
||||||
<img
|
|
||||||
src="/images/icon-pl.svg"
|
|
||||||
alt="icon-pl"
|
|
||||||
@click="changeLang('en')"
|
|
||||||
v-if="currentLang == 'pl'"
|
|
||||||
/>
|
|
||||||
<img src="/images/icon-en.jpg" alt="icon-en" @click="changeLang('pl')" v-else />
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="header_body">
|
<div class="header_body">
|
||||||
<StatusIndicator />
|
<StatusIndicator />
|
||||||
|
|
||||||
<span class="header_brand">
|
<span class="header_brand">
|
||||||
<router-link to="/">
|
<router-link to="/">
|
||||||
<img src="/images/stacjownik-header-logo.svg" alt="Stacjownik" />
|
<img
|
||||||
|
v-if="isChristmas"
|
||||||
|
src="/images/stacjownik-header-logo-christmas.svg"
|
||||||
|
alt="Stacjownik logo (christmas)"
|
||||||
|
/>
|
||||||
|
<img v-else src="/images/stacjownik-header-logo.svg" alt="Stacjownik logo" />
|
||||||
</router-link>
|
</router-link>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
@@ -40,17 +33,17 @@
|
|||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="header_links">
|
<span class="header_links">
|
||||||
<router-link class="route" active-class="route-active" to="/" exact>
|
<router-link class="route-link" active-class="route-link-active" to="/" exact>
|
||||||
{{ $t('app.sceneries') }}
|
{{ $t('app.sceneries') }}
|
||||||
</router-link>
|
</router-link>
|
||||||
/
|
/
|
||||||
<router-link class="route" active-class="route-active" to="/trains">{{
|
<router-link class="route-link" active-class="route-link-active" to="/trains">{{
|
||||||
$t('app.trains')
|
$t('app.trains')
|
||||||
}}</router-link>
|
}}</router-link>
|
||||||
/
|
/
|
||||||
<router-link
|
<router-link
|
||||||
class="route"
|
class="route-link"
|
||||||
active-class="route-active"
|
active-class="route-link-active"
|
||||||
:data-active="$route.path.startsWith('/journal')"
|
:data-active="$route.path.startsWith('/journal')"
|
||||||
to="/journal"
|
to="/journal"
|
||||||
>
|
>
|
||||||
@@ -69,13 +62,7 @@ import Clock from './Clock.vue';
|
|||||||
import RegionDropdown from '../Global/RegionDropdown.vue';
|
import RegionDropdown from '../Global/RegionDropdown.vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
emits: ['changeLang'],
|
components: { StatusIndicator, Clock, RegionDropdown },
|
||||||
props: {
|
|
||||||
currentLang: {
|
|
||||||
type: String,
|
|
||||||
required: true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
setup() {
|
setup() {
|
||||||
return {
|
return {
|
||||||
@@ -83,12 +70,6 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
|
||||||
changeLang(lang: string) {
|
|
||||||
this.$emit('changeLang', lang);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
onlineTrainsCount() {
|
onlineTrainsCount() {
|
||||||
return this.store.trainList.filter((train) => train.region == this.store.region.id).length;
|
return this.store.trainList.filter((train) => train.region == this.store.region.id).length;
|
||||||
@@ -98,14 +79,19 @@ export default defineComponent({
|
|||||||
return this.store.activeSceneryList.filter(
|
return this.store.activeSceneryList.filter(
|
||||||
(scenery) => scenery.region == this.store.region.id && scenery.dispatcherId != -1
|
(scenery) => scenery.region == this.store.region.id && scenery.dispatcherId != -1
|
||||||
).length;
|
).length;
|
||||||
}
|
|
||||||
},
|
},
|
||||||
components: { StatusIndicator, Clock, RegionDropdown }
|
|
||||||
|
isChristmas() {
|
||||||
|
const date = new Date();
|
||||||
|
|
||||||
|
return date.getUTCMonth() == 11 && date.getUTCDate() >= 6 && date.getUTCDate() <= 31;
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/variables.scss';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/responsive.scss';
|
|
||||||
|
|
||||||
// HEADER
|
// HEADER
|
||||||
.app_header {
|
.app_header {
|
||||||
@@ -113,7 +99,7 @@ export default defineComponent({
|
|||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
|
||||||
position: relative;
|
position: relative;
|
||||||
background-color: $primaryCol;
|
background-color: #2c2c2c;
|
||||||
}
|
}
|
||||||
|
|
||||||
.header {
|
.header {
|
||||||
@@ -128,7 +114,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
border-radius: 0 0 1em 1em;
|
border-radius: 0 0 1em 1em;
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen {
|
||||||
position: relative;
|
position: relative;
|
||||||
margin-top: 0.5em;
|
margin-top: 0.5em;
|
||||||
}
|
}
|
||||||
@@ -167,20 +153,12 @@ export default defineComponent({
|
|||||||
|
|
||||||
padding: 0.5em;
|
padding: 0.5em;
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen {
|
||||||
transform: translateX(85%);
|
transform: translateX(85%);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ICONS
|
|
||||||
.icons-top {
|
|
||||||
img {
|
|
||||||
width: 2.5em;
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// COUNTER
|
// COUNTER
|
||||||
.info_counter {
|
.info_counter {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|||||||
@@ -0,0 +1,37 @@
|
|||||||
|
<template>
|
||||||
|
<div class="app-domain-info">
|
||||||
|
<div>
|
||||||
|
<img src="/images/icon-loading.svg" alt="loading" height="200" />
|
||||||
|
<h1><span class="text--primary">Aplikacja</span> została przeniesiona na nową domenę!</h1>
|
||||||
|
<h1><span class="text--primary">This app</span> has been moved to a new domain!</h1>
|
||||||
|
|
||||||
|
<div style="margin-top: 1em">
|
||||||
|
<a :href="newLink">Nowy link dla obecnego adresu / New link to the current address</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import { computed } from 'vue';
|
||||||
|
|
||||||
|
const newLink = computed(() => {
|
||||||
|
return 'https://stacjownik-td2.spythere.eu' + location.pathname + location.search;
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.app-domain-info {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
text-align: center;
|
||||||
|
min-height: 100vh;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
font-size: 1.35em;
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,236 @@
|
|||||||
|
<template>
|
||||||
|
<Card :is-open="props.isCardOpen">
|
||||||
|
<div class="body-content">
|
||||||
|
<h1>{{ $t('welcome.title') }}</h1>
|
||||||
|
|
||||||
|
<div class="language-select">
|
||||||
|
<button :data-active="$i18n.locale == 'pl'" @click="store.changeLocale('pl')">
|
||||||
|
<img src="/images/icon-pl.svg" alt="" width="45" />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button :data-active="$i18n.locale == 'en'" @click="store.changeLocale('en')">
|
||||||
|
<img src="/images/icon-en.svg" alt="" width="45" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<section class="app-description">
|
||||||
|
<i18n-t keypath="welcome.app-desc" tag="p">
|
||||||
|
<template v-slot:b1>
|
||||||
|
<b>{{ $t('welcome.app-desc-b1') }}</b>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<template v-slot:link>
|
||||||
|
<a href="https://td2.info.pl/" class="link" target="_blank">Train Driver 2</a>
|
||||||
|
</template>
|
||||||
|
</i18n-t>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="tabs">
|
||||||
|
<div class="tab-description">
|
||||||
|
<h2 class="text--primary">{{ $t('welcome.sceneries-header') }}</h2>
|
||||||
|
<hr />
|
||||||
|
<i18n-t keypath="welcome.sceneries-desc" tag="p">
|
||||||
|
<template v-slot:b1>
|
||||||
|
<b>{{ $t('welcome.sceneries-desc-b1') }}</b>
|
||||||
|
</template>
|
||||||
|
</i18n-t>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="tab-description">
|
||||||
|
<h2 class="text--primary">{{ $t('welcome.trains-header') }}</h2>
|
||||||
|
<hr />
|
||||||
|
<i18n-t keypath="welcome.trains-desc" tag="p">
|
||||||
|
<template v-slot:b1>
|
||||||
|
<b>{{ $t('welcome.trains-desc-b1') }}</b>
|
||||||
|
</template>
|
||||||
|
</i18n-t>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="tab-description">
|
||||||
|
<h2 class="text--primary">{{ $t('welcome.journal-header') }}</h2>
|
||||||
|
<hr />
|
||||||
|
<i18n-t keypath="welcome.journal-desc" tag="p">
|
||||||
|
<template v-slot:b1>
|
||||||
|
<b>{{ $t('welcome.journal-desc-b1') }}</b>
|
||||||
|
</template>
|
||||||
|
</i18n-t>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="other-apps">
|
||||||
|
<b class="text--primary">
|
||||||
|
{{ $t('welcome.other-apps') }}
|
||||||
|
</b>
|
||||||
|
|
||||||
|
<div class="apps-grid">
|
||||||
|
<a class="app-item" href="https://pojazdownik-td2.web.app/" target="_blank">
|
||||||
|
<img src="/images/icon-pojazdownik.svg" alt="pojazdownik app logo" />
|
||||||
|
<h3 class="text--primary">Pojazdownik</h3>
|
||||||
|
<p>{{ $t('welcome.pojazdownik-desc') }}</p>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<a class="app-item" href="https://generator-td2.web.app/" target="_blank">
|
||||||
|
<img src="/images/icon-gnr.svg" alt="generator app logo" />
|
||||||
|
<h3 class="text--primary">GeneraTOR</h3>
|
||||||
|
<p>{{ $t('welcome.generator-desc') }}</p>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<a class="app-item" href="https://srjp-td2.web.app/" target="_blank">
|
||||||
|
<img src="/images/icon-srjp.svg" alt="srjp app logo" />
|
||||||
|
<h3 class="text--primary">Rozkładownik</h3>
|
||||||
|
<p>{{ $t('welcome.srjp-desc') }}</p>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="bottom-info">
|
||||||
|
<i18n-t keypath="welcome.donation-info" tag="div" class="donation-info">
|
||||||
|
<template v-slot:icon1>
|
||||||
|
<img src="/images/icon-diamond.svg" alt="diamond icon" width="25" />
|
||||||
|
<span class="text--donator"> {{ $t('welcome.donation-info-icon1-text') }}</span>
|
||||||
|
</template>
|
||||||
|
</i18n-t>
|
||||||
|
|
||||||
|
<i18n-t keypath="welcome.discord-info" tag="div" class="discord-info">
|
||||||
|
<template v-slot:discord>
|
||||||
|
<a href="https://discord.gg/x2mpNN3svk" class="link" target="_blank">
|
||||||
|
<b class="text--discord">{{ $t('welcome.discord-info-link-text') }}</b>
|
||||||
|
</a>
|
||||||
|
</template>
|
||||||
|
</i18n-t>
|
||||||
|
|
||||||
|
<div class="bottom-text">
|
||||||
|
<i>{{ $t('welcome.bottom-text') }}</i>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="bottom-actions">
|
||||||
|
<button class="btn btn--action" @click="toggleCard(false)">
|
||||||
|
{{ $t('welcome.button-confirm') }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
</Card>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import Card from '../Global/Card.vue';
|
||||||
|
import { useMainStore } from '../../store/mainStore';
|
||||||
|
|
||||||
|
const store = useMainStore();
|
||||||
|
|
||||||
|
const emit = defineEmits(['toggleCard']);
|
||||||
|
const props = defineProps({
|
||||||
|
isCardOpen: Boolean
|
||||||
|
});
|
||||||
|
|
||||||
|
function toggleCard(state: boolean) {
|
||||||
|
emit('toggleCard', state);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.body-content {
|
||||||
|
max-width: 800px;
|
||||||
|
min-height: 900px;
|
||||||
|
padding: 1em 0.5em;
|
||||||
|
|
||||||
|
text-align: center;
|
||||||
|
font-size: 1.1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
margin-bottom: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
a.link {
|
||||||
|
text-decoration: underline;
|
||||||
|
|
||||||
|
img {
|
||||||
|
vertical-align: middle;
|
||||||
|
margin-right: 0.2em;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.language-select {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
margin: 0.5em 0;
|
||||||
|
|
||||||
|
button[data-active='false'] img {
|
||||||
|
opacity: 0.5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.app-description {
|
||||||
|
margin: 1em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-description {
|
||||||
|
margin-top: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.other-apps {
|
||||||
|
font-weight: bold;
|
||||||
|
margin: 1em 0;
|
||||||
|
font-size: 1.1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.apps-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
|
||||||
|
gap: 1em;
|
||||||
|
padding: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.apps-grid > a.app-item {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5em;
|
||||||
|
padding: 1em;
|
||||||
|
background-color: #2b2b2b;
|
||||||
|
transition: background-color 100ms ease-in-out;
|
||||||
|
border-radius: 0.5em;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: #3b3b3b;
|
||||||
|
}
|
||||||
|
|
||||||
|
img {
|
||||||
|
width: 2.5em;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.donation-info {
|
||||||
|
font-weight: bold;
|
||||||
|
font-size: 1.1em;
|
||||||
|
|
||||||
|
img {
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.discord-info {
|
||||||
|
margin-top: 1em;
|
||||||
|
font-weight: bold;
|
||||||
|
|
||||||
|
img {
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.bottom-text {
|
||||||
|
margin: 1em 0;
|
||||||
|
font-weight: bold;
|
||||||
|
font-size: 1.2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bottom-actions {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
margin-top: 1em;
|
||||||
|
|
||||||
|
font-size: 1.25em;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -6,9 +6,7 @@
|
|||||||
import { computed, defineComponent, ref } from 'vue';
|
import { computed, defineComponent, ref } from 'vue';
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
name: 'VueClock',
|
name: 'VueClock',
|
||||||
data: () => ({
|
data: () => ({ timestamp: Date.now() }),
|
||||||
timestamp: Date.now()
|
|
||||||
}),
|
|
||||||
setup() {
|
setup() {
|
||||||
let timestamp = ref(Date.now());
|
let timestamp = ref(Date.now());
|
||||||
|
|
||||||
@@ -28,8 +26,6 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
|
||||||
|
|
||||||
.clock {
|
.clock {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
|||||||
@@ -0,0 +1,94 @@
|
|||||||
|
<template>
|
||||||
|
<Card :is-open="isOpen" @toggle-card="toggleCard">
|
||||||
|
<div class="body-content">
|
||||||
|
<div class="content-top">
|
||||||
|
<img src="/images/icon-loading.svg" alt="loading" height="125" />
|
||||||
|
<h1>{{ t('migrate-info.header-text') }}</h1>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<p v-html="t('migrate-info.paragraph-1-html')"></p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
<a class="new-link" href="https://stacjownik-td2.spythere.eu/" target="_blank">
|
||||||
|
{{ t('migrate-info.paragraph-2-link-text') }}
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
{{ t('migrate-info.paragraph-3-text') }}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p class="info-bottom" v-html="t('migrate-info.paragraph-4-html')"></p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="content-actions">
|
||||||
|
<button class="btn btn--action" @click="toggleCard">PRZYJĄŁEM!</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Card>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import Card from '../Global/Card.vue';
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
defineProps({
|
||||||
|
isOpen: {
|
||||||
|
type: Boolean,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const emit = defineEmits(['toggleCard']);
|
||||||
|
|
||||||
|
function toggleCard() {
|
||||||
|
emit('toggleCard');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.body-content {
|
||||||
|
max-width: 800px;
|
||||||
|
min-height: 500px;
|
||||||
|
padding: 1em 0.5em;
|
||||||
|
|
||||||
|
display: grid;
|
||||||
|
grid-template-rows: auto 1fr auto;
|
||||||
|
gap: 0.5em;
|
||||||
|
|
||||||
|
text-align: center;
|
||||||
|
font-size: 1.1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
padding: 0.5em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
a.new-link {
|
||||||
|
font-size: 1.2em;
|
||||||
|
color: var(--clr-primary);
|
||||||
|
color: transparent;
|
||||||
|
|
||||||
|
background: var(--clr-primary);
|
||||||
|
background: linear-gradient(90deg, var(--clr-primary), #ffffff);
|
||||||
|
background-clip: text;
|
||||||
|
-webkit-background-clip: text;
|
||||||
|
-webkit-text-fill-color: transparent;
|
||||||
|
|
||||||
|
text-shadow: var(--clr-primary) 0 0 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-bottom {
|
||||||
|
font-size: 0.9em;
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content-actions {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
font-size: 1.2em;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -310,7 +310,7 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
// INDICATOR TOOLTIP ANIMATION
|
// INDICATOR TOOLTIP ANIMATION
|
||||||
.tooltip-anim {
|
.tooltip-anim {
|
||||||
@@ -379,7 +379,7 @@ export default defineComponent({
|
|||||||
content: '';
|
content: '';
|
||||||
}
|
}
|
||||||
|
|
||||||
@include midScreen() {
|
@include responsive.midScreen() {
|
||||||
left: auto;
|
left: auto;
|
||||||
right: 200%;
|
right: 200%;
|
||||||
|
|
||||||
@@ -393,7 +393,7 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen() {
|
@include responsive.smallScreen{
|
||||||
min-width: 8em;
|
min-width: 8em;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -73,11 +73,9 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/variables';
|
|
||||||
|
|
||||||
::v-deep(h1) {
|
::v-deep(h1) {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
color: $accentCol;
|
color: var(--clr-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
::v-deep(h2) {
|
::v-deep(h2) {
|
||||||
|
|||||||
@@ -0,0 +1,81 @@
|
|||||||
|
<template>
|
||||||
|
<div class="driver-not-found">
|
||||||
|
<h2>⦻ {{ $t('trains.driver-not-found-header') }}</h2>
|
||||||
|
|
||||||
|
<p class="text--grayed">
|
||||||
|
{{ $t('trains.driver-not-found-desc-1') }} <br />
|
||||||
|
{{ $t('trains.driver-not-found-desc-2') }}
|
||||||
|
<router-link to="/journal/timetables"
|
||||||
|
>{{ $t('trains.driver-not-found-journal') }} </router-link
|
||||||
|
>!
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p v-if="props.trainId && otherDriverTrains.length > 0">
|
||||||
|
<i18n-t keypath="trains.driver-not-found-others">
|
||||||
|
<template v-slot:driver>
|
||||||
|
<b>{{ otherDriverTrains[0].driverName }}</b>
|
||||||
|
</template>
|
||||||
|
</i18n-t>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="other-driver-trains">
|
||||||
|
<template v-for="(train, i) in otherDriverTrains">
|
||||||
|
<router-link :to="`/driver?trainId=${train.id}`">
|
||||||
|
{{ train.trainNo }}
|
||||||
|
| {{ regions.find((r) => r.id == train.region)?.name ?? 'PL1' }}
|
||||||
|
</router-link>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div style="margin-top: 1em">
|
||||||
|
<router-link to="/"><< {{ $t('trains.driver-not-found-return') }}</router-link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { computed } from 'vue';
|
||||||
|
import { useMainStore } from '../../store/mainStore';
|
||||||
|
import { regions } from '../../data/options.json';
|
||||||
|
|
||||||
|
const mainStore = useMainStore();
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
trainId: {
|
||||||
|
type: String
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const otherDriverTrains = computed(() => {
|
||||||
|
return mainStore.trainList.filter(
|
||||||
|
(train) =>
|
||||||
|
train.driverId == Number(props.trainId?.split('|')[0]) &&
|
||||||
|
(train.timetableData || train.online || train.lastSeen >= Date.now() - 60000)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.driver-not-found {
|
||||||
|
background-color: var(--clr-view-bg);
|
||||||
|
text-align: center;
|
||||||
|
padding: 1em;
|
||||||
|
border-radius: 0.5em 0.5em;
|
||||||
|
|
||||||
|
p {
|
||||||
|
padding: 0.5em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
text-decoration: underline;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.other-driver-trains {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,85 @@
|
|||||||
|
<template>
|
||||||
|
<div class="driver-top-actions">
|
||||||
|
<div class="actions-container">
|
||||||
|
<div class="actions actions-left">
|
||||||
|
<button class="a-button btn--filled btn--image" @click="routerReturn">
|
||||||
|
<img src="/images/icon-back.svg" alt="train icon" />
|
||||||
|
<span>
|
||||||
|
{{ t('trains.driver-return-link') }}
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="actions actions-right">
|
||||||
|
<a class="a-button btn--filled btn--image" :href="`https://srjp-td2.web.app/?id=${chosenTrain.id}`"
|
||||||
|
target="_blank">
|
||||||
|
<span class="hidable">
|
||||||
|
{{ t('trains.driver-srjp-link') }}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<img src="/images/icon-srjp.svg" alt="srjp icon" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<router-link :to="`/journal/timetables?search-driver=${chosenTrain.driverName}`"
|
||||||
|
class="a-button btn--filled btn--image">
|
||||||
|
<span class="hidable">
|
||||||
|
{{ t('trains.driver-journal-link') }}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<img src="/images/icon-train.svg" alt="train icon" />
|
||||||
|
</router-link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useRouter } from 'vue-router';
|
||||||
|
import { Train } from '../../typings/common';
|
||||||
|
import { PropType } from 'vue';
|
||||||
|
|
||||||
|
const router = useRouter();
|
||||||
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
defineProps({
|
||||||
|
chosenTrain: {
|
||||||
|
type: Object as PropType<Train>,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function routerReturn() {
|
||||||
|
router.back();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
|
|
||||||
|
.actions-container {
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-end;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actions {
|
||||||
|
display: flex;
|
||||||
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.actions-container>.actions>.a-button {
|
||||||
|
padding: 0.5em;
|
||||||
|
border-radius: 0.5em 0.5em 0 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@include responsive.smallScreen {
|
||||||
|
span.hidable {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,328 @@
|
|||||||
|
<template>
|
||||||
|
<div class="driver-train-card">
|
||||||
|
<TrainInfo :train="chosenTrain" :extended="true" />
|
||||||
|
|
||||||
|
<!-- Train action buttons -->
|
||||||
|
<div class="train-stock-actions">
|
||||||
|
<button class="btn btn--action" style="margin: 1em 0" @click="copyStockToClipboard()">
|
||||||
|
<i class="fa-regular fa-copy"></i> {{ i18n.t('trains.stock-copy') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<button class="btn btn--action" style="margin: 1em 0" @click="toggleNumberPropositions()">
|
||||||
|
<i class="fa-regular fa-lightbulb"></i> {{ i18n.t('trains.number-propositions') }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Proposed numbers container -->
|
||||||
|
<transition name="view-anim" class="propositions-container">
|
||||||
|
<div v-if="arePropositionsVisible">
|
||||||
|
<h3 style="margin-bottom: 0.5em">{{ i18n.t('trains.number-propositions-header') }}</h3>
|
||||||
|
|
||||||
|
<div class="categories-select">
|
||||||
|
<button
|
||||||
|
v-for="(category, i) in availableCategories"
|
||||||
|
class="btn btn--option btn--action"
|
||||||
|
@click="selectCategory(i)"
|
||||||
|
:class="{ checked: i == chosenCategoryIndex }"
|
||||||
|
>
|
||||||
|
{{ category }}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="numberPropositions.length > 0" class="propositions-numbers">
|
||||||
|
<div v-if="chosenCategory">
|
||||||
|
<b>{{ chosenCategory }} </b> -
|
||||||
|
{{ i18n.t(`categories.${chosenCategory.slice(0, 2)}`) }}
|
||||||
|
({{ i18n.t(`categories.${chosenCategory.slice(2)}`) }})
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-if="chosenCategoryRules">
|
||||||
|
<span v-if="chosenCategoryRules[0]"
|
||||||
|
>{{ i18n.t('trains.number-propositions-third-number') }}
|
||||||
|
<b class="text--primary">{{ chosenCategoryRules[0] }}</b> •
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span
|
||||||
|
>{{
|
||||||
|
i18n.t('trains.number-propositions-last-nums', {
|
||||||
|
count: chosenCategoryRules[1].length
|
||||||
|
})
|
||||||
|
}}
|
||||||
|
<b class="text--primary">{{ chosenCategoryRules[1] }}</b> -
|
||||||
|
<b class="text--primary">{{ chosenCategoryRules[2] }}</b></span
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div style="margin-top: 0.5em">
|
||||||
|
<b>{{ i18n.t('trains.number-propositions-title') }} </b>
|
||||||
|
<i>{{ numberPropositions.join(', ') }}</i>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="no-propositions" v-else>{{ i18n.t('trains.number-propositions-empty') }}</div>
|
||||||
|
</div>
|
||||||
|
</transition>
|
||||||
|
|
||||||
|
<StockList :trainStockList="chosenTrain.stockList" />
|
||||||
|
<TrainSchedule :train="chosenTrain" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { PropType, ref } from 'vue';
|
||||||
|
import { Train } from '../../typings/common';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { useApiStore } from '../../store/apiStore';
|
||||||
|
|
||||||
|
import StockList from '../Global/StockList.vue';
|
||||||
|
import TrainSchedule from '../TrainsView/TrainSchedule.vue';
|
||||||
|
import TrainInfo from '../TrainsView/TrainInfo.vue';
|
||||||
|
|
||||||
|
import rulesJSON from '../../data/trainNumberRules.json';
|
||||||
|
import { computed } from 'vue';
|
||||||
|
import { watch } from 'vue';
|
||||||
|
|
||||||
|
const apiStore = useApiStore();
|
||||||
|
|
||||||
|
const i18n = useI18n();
|
||||||
|
|
||||||
|
const arePropositionsVisible = ref(false);
|
||||||
|
const chosenCategoryIndex = ref(0);
|
||||||
|
|
||||||
|
const numberPropositions = ref<string[]>([]);
|
||||||
|
const chosenCategoryRules = ref<any[]>([]);
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
chosenTrain: {
|
||||||
|
type: Object as PropType<Train>,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
function copyStockToClipboard() {
|
||||||
|
const stockString = props.chosenTrain.stockList.join(';');
|
||||||
|
|
||||||
|
if (!stockString) {
|
||||||
|
alert(i18n.t('trains.stock-clipboard-failure'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
navigator.clipboard
|
||||||
|
.writeText(stockString)
|
||||||
|
.then(() => {
|
||||||
|
prompt(i18n.t('trains.stock-clipboard-success'), stockString);
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
alert(i18n.t('trains.stock-clipboard-failure'));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleNumberPropositions() {
|
||||||
|
arePropositionsVisible.value = !arePropositionsVisible.value;
|
||||||
|
|
||||||
|
if (arePropositionsVisible.value) generateNumberPropositions();
|
||||||
|
}
|
||||||
|
|
||||||
|
function selectCategory(i: number) {
|
||||||
|
chosenCategoryIndex.value = i;
|
||||||
|
|
||||||
|
generateNumberPropositions();
|
||||||
|
}
|
||||||
|
|
||||||
|
function generateNumberPropositions() {
|
||||||
|
const categoryCode = chosenCategory.value?.slice(0, 2);
|
||||||
|
const trainNoStr = props.chosenTrain.trainNo.toString();
|
||||||
|
|
||||||
|
// Get category rules
|
||||||
|
const rules = categoryCode
|
||||||
|
? ((rulesJSON.categoriesRules as any)[categoryCode] as any[])
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
if (!categoryCode || !rules) {
|
||||||
|
numberPropositions.value.length = 0;
|
||||||
|
chosenCategoryRules.value.length = 0;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const [thirdNumber, minRange, maxRange] = rules;
|
||||||
|
|
||||||
|
const propositionsArr: string[] = [];
|
||||||
|
|
||||||
|
for (let i = 0; i < 5; i++) {
|
||||||
|
let generatedNumStr = '';
|
||||||
|
|
||||||
|
generatedNumStr += trainNoStr.at(0) ?? Math.floor(Math.random() * 10);
|
||||||
|
generatedNumStr += trainNoStr.at(1) ?? Math.floor(Math.random() * 10);
|
||||||
|
|
||||||
|
// Third number
|
||||||
|
generatedNumStr += thirdNumber ?? '';
|
||||||
|
|
||||||
|
// Remaining numbers
|
||||||
|
const rangeNums = minRange?.length ?? 3;
|
||||||
|
|
||||||
|
const randRange = Math.floor(
|
||||||
|
Math.random() * (Number(maxRange) - Number(minRange)) + Number(minRange)
|
||||||
|
).toString();
|
||||||
|
|
||||||
|
const leadingZeros = new Array(Math.abs(randRange.toString().length - rangeNums))
|
||||||
|
.fill('0')
|
||||||
|
.join('');
|
||||||
|
|
||||||
|
generatedNumStr += `${leadingZeros}${randRange}`;
|
||||||
|
|
||||||
|
const isNumberTaken =
|
||||||
|
apiStore.activeData?.trains?.some((t) => t.trainNo.toString() == generatedNumStr) ?? false;
|
||||||
|
|
||||||
|
if (!isNumberTaken) {
|
||||||
|
propositionsArr.push(generatedNumStr);
|
||||||
|
} else {
|
||||||
|
i--;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Number(randRange) > Number(maxRange)) break;
|
||||||
|
}
|
||||||
|
|
||||||
|
numberPropositions.value = propositionsArr;
|
||||||
|
chosenCategoryRules.value = rules;
|
||||||
|
}
|
||||||
|
|
||||||
|
const chosenCategory = computed(() => {
|
||||||
|
return availableCategories.value.at(chosenCategoryIndex.value);
|
||||||
|
});
|
||||||
|
|
||||||
|
const availableCategories = computed(() => {
|
||||||
|
const stockList = props.chosenTrain.stockList;
|
||||||
|
const headVehicle = stockList.at(0)?.split('-')[0] ?? '';
|
||||||
|
|
||||||
|
let availableCategories: string[] = [];
|
||||||
|
let categoryTraction = 'E';
|
||||||
|
|
||||||
|
let vehicleTypesSet = new Set<string>();
|
||||||
|
let wagonsNamesSet = new Set<string>();
|
||||||
|
let cargoNamesSet = new Set<string>();
|
||||||
|
|
||||||
|
for (const stockName of stockList) {
|
||||||
|
const [vehicleName, ...cargoList] = stockName.split(':');
|
||||||
|
|
||||||
|
const vehicleData = apiStore.vehiclesData?.find((v) => v.name == vehicleName);
|
||||||
|
|
||||||
|
if (!vehicleData) continue;
|
||||||
|
|
||||||
|
vehicleTypesSet.add(vehicleData.type);
|
||||||
|
|
||||||
|
if (vehicleData.type.startsWith('wagon-')) wagonsNamesSet.add(vehicleData.name.split('_')[0]);
|
||||||
|
|
||||||
|
if (cargoList !== undefined) cargoList.forEach((c) => cargoNamesSet.add(c.split('_')[0]));
|
||||||
|
}
|
||||||
|
|
||||||
|
let vehicleTypesArr = [...vehicleTypesSet];
|
||||||
|
let wagonsNamesArr = [...wagonsNamesSet];
|
||||||
|
|
||||||
|
// Traction
|
||||||
|
if (vehicleTypesArr[0] == 'loco-electric') categoryTraction = 'E';
|
||||||
|
else if (vehicleTypesArr[0] == 'loco-diesel') categoryTraction = 'S';
|
||||||
|
else if (vehicleTypesArr[0] == 'unit-electric') categoryTraction = 'J';
|
||||||
|
else categoryTraction = 'M';
|
||||||
|
|
||||||
|
// EMU / DMU - M*, R*, P*
|
||||||
|
if (vehicleTypesArr.length == 1 && (categoryTraction == 'J' || categoryTraction == 'M')) {
|
||||||
|
availableCategories.push('MO', 'MP', 'MM', 'RO', 'RP', 'RA', 'RM', 'PW');
|
||||||
|
}
|
||||||
|
// Only locos (up to 3) - LT, LP, LS
|
||||||
|
else if (stockList.length <= 3 && vehicleTypesArr.every((v) => v.startsWith('loco-'))) {
|
||||||
|
if (/^(EU|ET|201E|4E|SU|ST|M62|CTLR4C)/.test(headVehicle)) availableCategories.push('LT');
|
||||||
|
if (/^(EU|EP|SU|SP)/.test(headVehicle)) availableCategories.push('LP');
|
||||||
|
if (/^(SM)/.test(headVehicle)) availableCategories.push('LS');
|
||||||
|
}
|
||||||
|
// Only locos (more than 3) - TH
|
||||||
|
else if (stockList.length > 3 && vehicleTypesArr.every((v) => v.startsWith('loco-'))) {
|
||||||
|
availableCategories.push('TH');
|
||||||
|
}
|
||||||
|
// Loco(s) + passenger only wagons - M*, R*, E*, P*
|
||||||
|
else if (vehicleTypesArr.every((v) => v.startsWith('loco-') || v == 'wagon-passenger')) {
|
||||||
|
availableCategories.push('EI', 'EC', 'EN', 'MO', 'MP', 'MM', 'RO', 'RP', 'RA', 'RM', 'PW');
|
||||||
|
}
|
||||||
|
// Loco(s) + cargo only / mixed wagons - T*, Z*
|
||||||
|
else {
|
||||||
|
if (wagonsNamesArr.every((v) => /^(627Z|412Z)/.test(v)))
|
||||||
|
availableCategories.push('TC', 'TD', 'TS');
|
||||||
|
else if (stockList.slice(1).every((v) => /PKPE/.test(v))) {
|
||||||
|
availableCategories.push('ZU', 'ZN');
|
||||||
|
} else if (wagonsNamesArr.length < 3 || cargoNamesSet.size < 3) {
|
||||||
|
availableCategories.push('TM', 'TG', 'TS', 'TK');
|
||||||
|
} else {
|
||||||
|
availableCategories.push('TN', 'TR', 'TS', 'TK');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return availableCategories.map((c) => `${c}${categoryTraction}`);
|
||||||
|
});
|
||||||
|
|
||||||
|
watch(
|
||||||
|
computed(() => `${props.chosenTrain.trainNo}`),
|
||||||
|
() => {
|
||||||
|
chosenCategoryIndex.value = 0;
|
||||||
|
generateNumberPropositions();
|
||||||
|
}
|
||||||
|
);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
|
.driver-train-card {
|
||||||
|
padding: 1em;
|
||||||
|
background-color: var(--clr-view-bg);
|
||||||
|
border-radius: 0 0 0.5em 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.train-stock-actions {
|
||||||
|
display: flex;
|
||||||
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.propositions-container {
|
||||||
|
margin-bottom: 1em;
|
||||||
|
padding: 0.5em;
|
||||||
|
background-color: #111;
|
||||||
|
}
|
||||||
|
|
||||||
|
.categories-select {
|
||||||
|
display: inline-flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 0.5em;
|
||||||
|
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
bottom: calc(-0.5em);
|
||||||
|
left: 0;
|
||||||
|
|
||||||
|
width: 100%;
|
||||||
|
height: 2px;
|
||||||
|
background-color: #aaa;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.propositions-numbers {
|
||||||
|
margin-top: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.no-propositions {
|
||||||
|
margin-top: 1em;
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
|
|
||||||
|
@include responsive.smallScreen {
|
||||||
|
.propositions-container {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.categories-select {
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -13,8 +13,7 @@ export default defineComponent({});
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
@import '../../styles/variables';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/responsive';
|
|
||||||
|
|
||||||
.button_content {
|
.button_content {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|||||||
@@ -38,5 +38,3 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped></style>
|
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
.card {
|
.card {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
@@ -81,13 +81,8 @@ export default defineComponent({
|
|||||||
|
|
||||||
background-color: #1a1a1a;
|
background-color: #1a1a1a;
|
||||||
box-shadow: 0 0 15px 10px #0e0e0e;
|
box-shadow: 0 0 15px 10px #0e0e0e;
|
||||||
|
border-radius: 1em;
|
||||||
|
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
|
||||||
.card {
|
|
||||||
align-items: flex-start;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
<transition mode="out-in" name="slider-anim" class="current-name">
|
<transition mode="out-in" name="slider-anim" class="current-name">
|
||||||
<span :key="displayingName">
|
<span :key="displayingName">
|
||||||
<img src="/images/icon-diamond.svg" alt="donator diamond icon" />
|
<img src="/images/icon-diamond.svg" alt="donator diamond icon" />
|
||||||
{{ displayingName }}
|
<span class="text--donator">{{ displayingName }}</span>
|
||||||
</span>
|
</span>
|
||||||
</transition>
|
</transition>
|
||||||
</div>
|
</div>
|
||||||
@@ -45,7 +45,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template v-slot:b2>
|
<template v-slot:b2>
|
||||||
<b>{{ $t('donations.p4-b2') }}</b>
|
<b class="text--donator">{{ $t('donations.p4-b2') }}</b>
|
||||||
</template>
|
</template>
|
||||||
</i18n-t>
|
</i18n-t>
|
||||||
<br />
|
<br />
|
||||||
@@ -56,7 +56,7 @@
|
|||||||
</i>
|
</i>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="actions">
|
<div class="actions-container">
|
||||||
<a
|
<a
|
||||||
class="action a-button btn--image coffee"
|
class="action a-button btn--image coffee"
|
||||||
href="https://buycoffee.to/spythere"
|
href="https://buycoffee.to/spythere"
|
||||||
@@ -150,8 +150,6 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
|
||||||
|
|
||||||
.body {
|
.body {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-rows: 1fr auto;
|
grid-template-rows: 1fr auto;
|
||||||
@@ -187,7 +185,7 @@ a.discord {
|
|||||||
text-decoration: underline;
|
text-decoration: underline;
|
||||||
}
|
}
|
||||||
|
|
||||||
.actions {
|
.actions-container {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
@@ -198,32 +196,28 @@ a.discord {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.actions > .action {
|
.actions-container > .action {
|
||||||
&.paypal {
|
&.paypal {
|
||||||
$btnColor: #254069;
|
background-color: #254069;
|
||||||
|
|
||||||
background-color: $btnColor;
|
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background-color: lighten($btnColor, 5%);
|
background-color: #2f5185;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&.coffee {
|
&.coffee {
|
||||||
$btnColor: #009255;
|
background-color: #009255;
|
||||||
background-color: $btnColor;
|
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background-color: lighten($btnColor, 5%);
|
background-color: #00a35f;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&.exit {
|
&.exit {
|
||||||
$btnColor: #686868;
|
background-color: #686868;
|
||||||
background-color: $btnColor;
|
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
background-color: lighten($btnColor, 5%);
|
background-color: #8d8d8d;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -120,8 +120,6 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/variables.scss';
|
|
||||||
|
|
||||||
.region-dropdown {
|
.region-dropdown {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -182,7 +180,7 @@ li.option {
|
|||||||
background: none;
|
background: none;
|
||||||
|
|
||||||
&:focus + span {
|
&:focus + span {
|
||||||
color: $accentCol;
|
color: var(--clr-primary);
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,7 +7,12 @@
|
|||||||
@keypress="updateValue"
|
@keypress="updateValue"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<img class="search-exit" src="/images/icon-exit.svg" alt="exit-icon" @click="clearSearchValue" />
|
<img
|
||||||
|
class="search-exit"
|
||||||
|
src="/images/icon-exit.svg"
|
||||||
|
alt="exit-icon"
|
||||||
|
@click="clearSearchValue"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -17,21 +22,10 @@ import { defineComponent, ref, watch } from 'vue';
|
|||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
emits: ['update:searchedValue', 'clearValue'],
|
emits: ['update:searchedValue', 'clearValue'],
|
||||||
props: {
|
props: {
|
||||||
searchedValue: {
|
searchedValue: { type: String, required: true },
|
||||||
type: String,
|
updateOnInput: { type: Boolean, default: true },
|
||||||
required: true
|
titleToTranslate: { type: String, required: true },
|
||||||
},
|
clearValue: { type: Function }
|
||||||
updateOnInput: {
|
|
||||||
type: Boolean,
|
|
||||||
default: true
|
|
||||||
},
|
|
||||||
titleToTranslate: {
|
|
||||||
type: String,
|
|
||||||
required: true
|
|
||||||
},
|
|
||||||
clearValue: {
|
|
||||||
type: Function
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
setup(props, { emit }) {
|
setup(props, { emit }) {
|
||||||
@@ -56,17 +50,13 @@ export default defineComponent({
|
|||||||
emit('update:searchedValue', compSearchedValue.value);
|
emit('update:searchedValue', compSearchedValue.value);
|
||||||
};
|
};
|
||||||
|
|
||||||
return {
|
return { compSearchedValue, updateValue, clearSearchValue };
|
||||||
compSearchedValue,
|
|
||||||
updateValue,
|
|
||||||
clearSearchValue
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive';
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
.search {
|
.search {
|
||||||
&-box {
|
&-box {
|
||||||
@@ -78,7 +68,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
margin: 0.5em 0 0.5em 0.5em;
|
margin: 0.5em 0 0.5em 0.5em;
|
||||||
|
|
||||||
@include smallScreen() {
|
@include responsive.smallScreen{
|
||||||
width: 85%;
|
width: 85%;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -20,15 +20,9 @@ import { Status } from '../../typings/common';
|
|||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
props: {
|
props: {
|
||||||
dispatcherStatus: {
|
dispatcherStatus: { type: Number as PropType<Status.ActiveDispatcher | number> },
|
||||||
type: Number as PropType<Status.ActiveDispatcher | number>
|
dispatcherTimestamp: { type: Number as PropType<number | null> },
|
||||||
},
|
isOnline: { type: Boolean }
|
||||||
dispatcherTimestamp: {
|
|
||||||
type: Number as PropType<number | null>
|
|
||||||
},
|
|
||||||
isOnline: {
|
|
||||||
type: Boolean
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
mixins: [dateMixin],
|
mixins: [dateMixin],
|
||||||
|
|
||||||
|
|||||||
@@ -1,25 +1,13 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="stock-list">
|
<div class="list-wrapper">
|
||||||
<ul>
|
<ul class="stock-list">
|
||||||
<li
|
<li v-for="({ images, imagesFallbacks, vehicleString }, i) in thumbnailNames">
|
||||||
v-for="(
|
|
||||||
{ vehicleName, vehicleCargo, images, imagesFallbacks, vehicleString }, i
|
|
||||||
) in thumbnailNames"
|
|
||||||
:key="i"
|
|
||||||
>
|
|
||||||
<div class="stock-text">
|
|
||||||
<div>{{ vehicleName.replace(/_/g, ' ') }}</div>
|
|
||||||
<small v-if="vehicleCargo">({{ vehicleCargo }})</small>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<span>
|
|
||||||
<VehicleThumbnail
|
<VehicleThumbnail
|
||||||
v-for="(thumbnailImage, imageIndex) in images"
|
:key="i"
|
||||||
:vehicle-name="vehicleString"
|
:vehicle-string="vehicleString"
|
||||||
:img-name="thumbnailImage"
|
:images="images"
|
||||||
:fallback-name="imagesFallbacks[imageIndex]"
|
:image-fallbacks="imagesFallbacks"
|
||||||
/>
|
/>
|
||||||
</span>
|
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
@@ -34,20 +22,12 @@ export default defineComponent({
|
|||||||
components: { VehicleThumbnail },
|
components: { VehicleThumbnail },
|
||||||
|
|
||||||
props: {
|
props: {
|
||||||
trainStockList: {
|
trainStockList: { type: Array as PropType<string[]>, required: true },
|
||||||
type: Array as PropType<string[]>,
|
tractionOnly: { type: Boolean, required: false }
|
||||||
required: true
|
|
||||||
},
|
|
||||||
tractionOnly: {
|
|
||||||
type: Boolean,
|
|
||||||
required: false
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return { apiStore: useApiStore() };
|
||||||
apiStore: useApiStore()
|
|
||||||
};
|
|
||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
@@ -59,13 +39,12 @@ export default defineComponent({
|
|||||||
return (this.tractionOnly ? this.trainStockList.slice(0, 1) : this.trainStockList)
|
return (this.tractionOnly ? this.trainStockList.slice(0, 1) : this.trainStockList)
|
||||||
.filter((v) => v.length != 0)
|
.filter((v) => v.length != 0)
|
||||||
.map((vehicleString) => {
|
.map((vehicleString) => {
|
||||||
const [vehicleName, vehicleCargo] = vehicleString.split(':');
|
const [vehicleName] = vehicleString.split(':');
|
||||||
|
|
||||||
const vehicleThumbnailData = {
|
const vehicleThumbnailData = {
|
||||||
images: [] as string[],
|
images: [] as string[],
|
||||||
imagesFallbacks: [] as string[],
|
imagesFallbacks: [] as string[],
|
||||||
vehicleName,
|
vehicleName,
|
||||||
vehicleCargo,
|
|
||||||
vehicleString
|
vehicleString
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -159,40 +138,20 @@ export default defineComponent({
|
|||||||
return vehicleThumbnailData;
|
return vehicleThumbnailData;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
|
||||||
|
|
||||||
methods: {
|
|
||||||
onImageError(event: Event, fallbackImage: string) {
|
|
||||||
(event.target as HTMLImageElement).src = `/images/${fallbackImage}.png`;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.stock-list {
|
.list-wrapper {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.stock-list ul {
|
.stock-list {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: flex-end;
|
align-items: flex-end;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
ul > li > span {
|
|
||||||
display: flex;
|
|
||||||
align-items: flex-end;
|
|
||||||
cursor: crosshair;
|
|
||||||
}
|
|
||||||
|
|
||||||
.stock-text {
|
|
||||||
text-align: center;
|
|
||||||
color: #aaa;
|
|
||||||
font-size: 0.9em;
|
|
||||||
margin-bottom: 0.25em;
|
|
||||||
padding: 0.25em 0;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,34 +1,42 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="vehicle-thumbnail" :data-load-status="imgStatus">
|
<div class="vehicle-thumbnail" :data-load-status="imgStatus" ref="thumbRef">
|
||||||
|
<div class="stock-text">
|
||||||
|
<div>{{ vehicleName }}</div>
|
||||||
|
<small v-if="vehicleCargo">({{ vehicleCargo }})</small>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="stock-images">
|
||||||
<img
|
<img
|
||||||
ref="imgRef"
|
v-for="(thumbnailImage, imageIndex) in images"
|
||||||
:src="`https://static.spythere.eu/thumbnails/v2/${imgName}.png`"
|
:src="`https://stacjownik.spythere.eu/static/thumbnails/${thumbnailImage}.png`"
|
||||||
height="60"
|
height="70"
|
||||||
loading="lazy"
|
loading="lazy"
|
||||||
data-tooltip-type="VehiclePreviewTooltip"
|
data-tooltip-type="VehiclePreviewTooltip"
|
||||||
:data-tooltip-content="vehicleName"
|
:data-tooltip-content="vehicleString"
|
||||||
@error="onImageError"
|
@error="onImageError($event, imageFallbacks[imageIndex])"
|
||||||
@load="onImageLoad"
|
@load="onImageLoad"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { Ref, ref } from 'vue';
|
import { computed, PropType, Ref, ref } from 'vue';
|
||||||
|
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
vehicleName: { type: String, required: true },
|
vehicleString: { type: String, required: true },
|
||||||
imgName: { type: String, required: true },
|
images: { type: Object as PropType<string[]>, required: true },
|
||||||
fallbackName: { type: String, required: true },
|
imageFallbacks: { type: Object as PropType<string[]>, required: true }
|
||||||
placeholderName: String
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const imgRef = ref(null) as Ref<HTMLElement | null>;
|
const thumbRef = ref(null) as Ref<HTMLElement | null>;
|
||||||
|
|
||||||
const imgStatus = ref('loading');
|
const imgStatus = ref('loading');
|
||||||
|
|
||||||
function onImageError(event: Event) {
|
const vehicleName = computed(() => props.vehicleString.split(':')[0].replace(/_/g, ' '));
|
||||||
(event.target as HTMLImageElement).src = `/images/${props.fallbackName}.png`;
|
const vehicleCargo = computed(() => props.vehicleString.split(':')[1]);
|
||||||
|
|
||||||
|
function onImageError(event: Event, fallbackImage: string) {
|
||||||
|
(event.target as HTMLImageElement).src = `/images/${fallbackImage}.png`;
|
||||||
imgStatus.value = 'error';
|
imgStatus.value = 'error';
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -37,22 +45,37 @@ function onImageLoad() {
|
|||||||
imgStatus.value = 'loaded';
|
imgStatus.value = 'loaded';
|
||||||
}
|
}
|
||||||
|
|
||||||
if (imgRef.value) imgRef.value.style.opacity = '1';
|
if (thumbRef.value) thumbRef.value.style.opacity = '1';
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.vehicle-thumbnail {
|
.vehicle-thumbnail {
|
||||||
position: relative;
|
position: relative;
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 100ms ease-in-out;
|
||||||
|
|
||||||
&[data-load-status='loading'] {
|
&[data-load-status='loading'] {
|
||||||
min-height: 60px;
|
min-height: 70px;
|
||||||
min-width: 200px;
|
min-width: 200px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
img {
|
.stock-text {
|
||||||
opacity: 0;
|
max-width: 90%;
|
||||||
transition: opacity 100ms ease-in-out;
|
text-align: center;
|
||||||
|
color: #aaa;
|
||||||
|
font-size: 0.85em;
|
||||||
|
margin: 0 auto;
|
||||||
|
padding: 0.25em 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.stock-images {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: flex-end;
|
||||||
|
cursor: crosshair;
|
||||||
|
|
||||||
|
padding: 0.5em 0;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -234,9 +234,9 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/animations';
|
||||||
@import '../../styles/JournalStats.scss';
|
@use '../../styles/journal-stats';
|
||||||
@import '../../styles/badge.scss';
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
.daily-stats {
|
.daily-stats {
|
||||||
text-align: left;
|
text-align: left;
|
||||||
@@ -265,7 +265,7 @@ ul.stats-list {
|
|||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen{
|
||||||
h3 {
|
h3 {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -121,14 +121,8 @@ import StationStatusBadge from '../../Global/StationStatusBadge.vue';
|
|||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
props: {
|
props: {
|
||||||
entry: {
|
entry: { type: Object as PropType<API.DispatcherHistory.Data>, required: true },
|
||||||
type: Object as PropType<API.DispatcherHistory.Data>,
|
showExtraInfo: { type: Boolean, required: true }
|
||||||
required: true
|
|
||||||
},
|
|
||||||
showExtraInfo: {
|
|
||||||
type: Boolean,
|
|
||||||
required: true
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
components: { StationStatusBadge },
|
components: { StationStatusBadge },
|
||||||
@@ -136,10 +130,7 @@ export default defineComponent({
|
|||||||
emits: ['toggleShowExtraInfo'],
|
emits: ['toggleShowExtraInfo'],
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return { regions, apiStore: useApiStore() };
|
||||||
regions,
|
|
||||||
apiStore: useApiStore()
|
|
||||||
};
|
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
@@ -151,8 +142,8 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/responsive.scss';
|
@use '../../../styles/responsive';
|
||||||
@import '../../../styles/badge.scss';
|
@use '../../../styles/badge';
|
||||||
|
|
||||||
.region-badge {
|
.region-badge {
|
||||||
padding: 0 0.25em;
|
padding: 0 0.25em;
|
||||||
@@ -207,7 +198,7 @@ export default defineComponent({
|
|||||||
border-radius: 1em;
|
border-radius: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen{
|
||||||
.entry-info {
|
.entry-info {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
|||||||
@@ -32,25 +32,25 @@
|
|||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<hr class="section-separator" />
|
<hr class="section-separator" v-if="stats.issuedTimetables" />
|
||||||
|
|
||||||
<div class="info-stats">
|
<div class="info-stats" v-if="stats.issuedTimetables">
|
||||||
<span class="badge stat-badge" v-if="stats.issuedTimetables">
|
<span class="badge stat-badge">
|
||||||
<span>{{ $t('journal.dispatcher-stats.timetables-count') }}</span>
|
<span>{{ $t('journal.dispatcher-stats.timetables-count') }}</span>
|
||||||
<span>{{ stats.issuedTimetables.count }}</span>
|
<span>{{ stats.issuedTimetables.count }}</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="badge stat-badge" v-if="stats.issuedTimetables">
|
<span class="badge stat-badge">
|
||||||
<span>{{ $t('journal.dispatcher-stats.timetables-sum') }}</span>
|
<span>{{ $t('journal.dispatcher-stats.timetables-sum') }}</span>
|
||||||
<span>{{ stats.issuedTimetables.distanceSum.toFixed(2) }}km</span>
|
<span>{{ stats.issuedTimetables.distanceSum.toFixed(2) }}km</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="badge stat-badge" v-if="stats.issuedTimetables">
|
<span class="badge stat-badge">
|
||||||
<span>{{ $t('journal.dispatcher-stats.timetables-max') }}</span>
|
<span>{{ $t('journal.dispatcher-stats.timetables-max') }}</span>
|
||||||
<span>{{ stats.issuedTimetables.distanceMax.toFixed(2) }}km</span>
|
<span>{{ stats.issuedTimetables.distanceMax.toFixed(2) }}km</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="badge stat-badge" v-if="stats.issuedTimetables">
|
<span class="badge stat-badge">
|
||||||
<span>{{ $t('journal.dispatcher-stats.timetables-avg') }}</span>
|
<span>{{ $t('journal.dispatcher-stats.timetables-avg') }}</span>
|
||||||
<span>{{ stats.issuedTimetables.distanceAvg.toFixed(2) }}km</span>
|
<span>{{ stats.issuedTimetables.distanceAvg.toFixed(2) }}km</span>
|
||||||
</span>
|
</span>
|
||||||
@@ -81,5 +81,5 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/JournalStats.scss';
|
@use '../../../styles/journal-stats';
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
|
<div>
|
||||||
<div class="journal_warning" v-if="store.isOffline">
|
<div class="journal_warning" v-if="store.isOffline">
|
||||||
{{ $t('app.offline') }}
|
{{ $t('app.offline') }}
|
||||||
</div>
|
</div>
|
||||||
@@ -39,6 +40,7 @@
|
|||||||
<div class="journal_warning" v-else-if="!scrollDataLoaded">
|
<div class="journal_warning" v-else-if="!scrollDataLoaded">
|
||||||
{{ $t('journal.loading-further-data') }}
|
{{ $t('journal.loading-further-data') }}
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
@@ -81,6 +83,15 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
|
watch: {
|
||||||
|
'$route.query': {
|
||||||
|
deep: true,
|
||||||
|
handler() {
|
||||||
|
this.extraInfoIndexes.length = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
toggleExtraInfo(id: number) {
|
toggleExtraInfo(id: number) {
|
||||||
const existingIdx = this.extraInfoIndexes.indexOf(id);
|
const existingIdx = this.extraInfoIndexes.indexOf(id);
|
||||||
@@ -93,6 +104,5 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/variables.scss';
|
@use '../../../styles/journal-section';
|
||||||
@import '../../../styles/JournalSection.scss';
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,11 +1,16 @@
|
|||||||
<template>
|
<template>
|
||||||
<section class="journal-header">
|
<section class="journal-header">
|
||||||
<div class="journal-type-options">
|
<div class="journal-type-options">
|
||||||
<router-link class="router-link" active-class="route-active" to="/journal/timetables" exact>
|
<router-link
|
||||||
|
class="route-link"
|
||||||
|
active-class="route-link-active"
|
||||||
|
to="/journal/timetables"
|
||||||
|
exact
|
||||||
|
>
|
||||||
{{ $t('journal.section-timetables') }}
|
{{ $t('journal.section-timetables') }}
|
||||||
</router-link>
|
</router-link>
|
||||||
•
|
•
|
||||||
<router-link class="router-link" active-class="route-active" to="/journal/dispatchers">
|
<router-link class="route-link" active-class="route-link-active" to="/journal/dispatchers">
|
||||||
{{ $t('journal.section-dispatchers') }}
|
{{ $t('journal.section-dispatchers') }}
|
||||||
</router-link>
|
</router-link>
|
||||||
</div>
|
</div>
|
||||||
@@ -39,8 +44,4 @@ export default defineComponent({});
|
|||||||
display: flex;
|
display: flex;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.router-link.active {
|
|
||||||
color: gold;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -33,7 +33,28 @@
|
|||||||
<h1 class="option-title">{{ $t('options.search-title') }}</h1>
|
<h1 class="option-title">{{ $t('options.search-title') }}</h1>
|
||||||
<div class="search_content">
|
<div class="search_content">
|
||||||
<div class="search" v-for="(_, propName) in searchersValues" :key="propName">
|
<div class="search" v-for="(_, propName) in searchersValues" :key="propName">
|
||||||
<label v-if="propName == 'search-date'" for="search-date">{{
|
<!-- Train category select -->
|
||||||
|
<div v-if="propName.toString() == 'select-categoryCode'">
|
||||||
|
<label for="journalCategoryCode">{{ $t(`options.${propName}`) }}</label>
|
||||||
|
|
||||||
|
<div class="search-box">
|
||||||
|
<select
|
||||||
|
class="search-input"
|
||||||
|
name="journalCategoryCode"
|
||||||
|
id="journalCategoryCode"
|
||||||
|
v-model="searchersValues[propName]"
|
||||||
|
>
|
||||||
|
<option value="">...</option>
|
||||||
|
<option v-for="categoryName in allCategories" :value="categoryName">
|
||||||
|
{{ categoryName }} - {{ getCategoryExplanation(categoryName) }}
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Other inputs -->
|
||||||
|
<div v-else>
|
||||||
|
<label v-if="propName == 'search-date-from'" for="search-date">{{
|
||||||
$t(`options.search-${optionsType}-date`)
|
$t(`options.search-${optionsType}-date`)
|
||||||
}}</label>
|
}}</label>
|
||||||
|
|
||||||
@@ -45,13 +66,16 @@
|
|||||||
@focus="preventKeyDown = true"
|
@focus="preventKeyDown = true"
|
||||||
@blur="preventKeyDown = false"
|
@blur="preventKeyDown = false"
|
||||||
:placeholder="$t(`options.${propName}`)"
|
:placeholder="$t(`options.${propName}`)"
|
||||||
:type="propName == 'search-date' ? 'date' : 'text'"
|
:type="propName.toString().startsWith('search-date') ? 'date' : 'text'"
|
||||||
:min="propName == 'search-date' ? '2022-02-01' : undefined"
|
:min="propName.toString().startsWith('search-date') ? '2022-02-01' : undefined"
|
||||||
:id="`${propName}`"
|
:id="`${propName}`"
|
||||||
:list="propName.toString()"
|
:list="propName.toString()"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<button class="search-exit" v-if="propName != 'search-date'">
|
<button
|
||||||
|
class="btn btn--action search-exit"
|
||||||
|
v-if="!propName.toString().startsWith('search-date')"
|
||||||
|
>
|
||||||
<img
|
<img
|
||||||
src="/images/icon-exit.svg"
|
src="/images/icon-exit.svg"
|
||||||
alt="exit-icon"
|
alt="exit-icon"
|
||||||
@@ -61,6 +85,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<h1 class="option-title">{{ $t('options.sort-title') }}</h1>
|
<h1 class="option-title">{{ $t('options.sort-title') }}</h1>
|
||||||
<div class="options_sorters">
|
<div class="options_sorters">
|
||||||
@@ -117,10 +142,12 @@ import { useMainStore } from '../../store/mainStore';
|
|||||||
import { Journal } from './typings';
|
import { Journal } from './typings';
|
||||||
import { Status } from '../../typings/common';
|
import { Status } from '../../typings/common';
|
||||||
import { useApiStore } from '../../store/apiStore';
|
import { useApiStore } from '../../store/apiStore';
|
||||||
|
import { allCategories } from '../../data/trainNumberRules.json';
|
||||||
|
import trainCategoryMixin from '../../mixins/trainCategoryMixin';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
emits: ['onSearchConfirm', 'onOptionsReset', 'onRefreshData'],
|
emits: ['onSearchConfirm', 'onOptionsReset', 'onRefreshData'],
|
||||||
mixins: [keyMixin],
|
mixins: [keyMixin, trainCategoryMixin],
|
||||||
|
|
||||||
props: {
|
props: {
|
||||||
sorterOptionIds: {
|
sorterOptionIds: {
|
||||||
@@ -152,6 +179,7 @@ export default defineComponent({
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
showOptions: false,
|
showOptions: false,
|
||||||
|
allCategories,
|
||||||
|
|
||||||
driverSuggestions: [] as string[],
|
driverSuggestions: [] as string[],
|
||||||
dispatcherSuggestions: [] as string[],
|
dispatcherSuggestions: [] as string[],
|
||||||
@@ -188,14 +216,14 @@ export default defineComponent({
|
|||||||
if (!value || value == '') return;
|
if (!value || value == '') return;
|
||||||
if (value.length < 3) return;
|
if (value.length < 3) return;
|
||||||
|
|
||||||
this.startSearchTimeout('driver', value);
|
if (this.showOptions) this.startSearchTimeout('driver', value);
|
||||||
},
|
},
|
||||||
|
|
||||||
async 'searchersValues.search-dispatcher'(value: string | undefined) {
|
async 'searchersValues.search-dispatcher'(value: string | undefined) {
|
||||||
if (!value || value == '') return;
|
if (!value || value == '') return;
|
||||||
if (value.length < 3) return;
|
if (value.length < 3) return;
|
||||||
|
|
||||||
this.startSearchTimeout('dispatcher', value);
|
if (this.showOptions) this.startSearchTimeout('dispatcher', value);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -283,7 +311,6 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
searchConfirm() {
|
searchConfirm() {
|
||||||
this.$emit('onSearchConfirm');
|
|
||||||
this.handleRouteParams();
|
this.handleRouteParams();
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -301,6 +328,6 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/dropdown';
|
@use '../../styles/dropdown';
|
||||||
@import '../../styles/dropdown_filters';
|
@use '../../styles/dropdown-filters';
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -79,14 +79,12 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/dropdown.scss';
|
@use '../../styles/dropdown';
|
||||||
@import '../../styles/dropdown_filters.scss';
|
@use '../../styles/dropdown-filters';
|
||||||
@import '../../styles/variables.scss';
|
|
||||||
|
|
||||||
.dropdown_wrapper.dropdown-align-right {
|
.dropdown_wrapper.dropdown-align-right {
|
||||||
left: auto;
|
left: auto;
|
||||||
right: 0;
|
right: 0;
|
||||||
max-width: 700px;
|
max-width: 700px;
|
||||||
// max-width: 100%;
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -23,18 +23,62 @@
|
|||||||
|
|
||||||
<div class="g-separator"></div>
|
<div class="g-separator"></div>
|
||||||
|
|
||||||
<div class="stock-specs">
|
<div class="timetable-specs">
|
||||||
<span class="badge" v-if="timetable.authorName">
|
<span class="badge specs-badge" v-if="timetable.authorName">
|
||||||
<span>{{ $t('journal.dispatcher-name') }}</span>
|
<span>{{ $t('journal.dispatcher-name') }}</span>
|
||||||
<span>{{ timetable.authorName }}</span>
|
<span>{{ timetable.authorName }}</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="badge" v-if="timetable.maxSpeed">
|
<span class="badge specs-badge" v-if="timetable.trainMaxSpeed">
|
||||||
|
<span>{{ $t('journal.stock-timetable-speed') }}</span>
|
||||||
|
<span> {{ timetable.trainMaxSpeed }}km/h </span>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span class="badge specs-badge" v-if="timetable.maxSpeed">
|
||||||
<span>{{ $t('journal.stock-max-speed') }}</span>
|
<span>{{ $t('journal.stock-max-speed') }}</span>
|
||||||
<span>{{ timetable.maxSpeed }}km/h</span>
|
<span>{{ timetable.maxSpeed }}km/h</span>
|
||||||
</span>
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
<span class="badge" v-if="timetable.stockLength">
|
<div class="stock-dangers" v-if="timetable.warningNotes">
|
||||||
|
<div class="g-separator"></div>
|
||||||
|
|
||||||
|
<b>{{ $t('journal.stock-dangers') }}:</b>
|
||||||
|
|
||||||
|
<ul>
|
||||||
|
<li v-if="timetable.twr">
|
||||||
|
<b class="text--primary">{{ $t('warnings.TWR') }} (TWR)</b>
|
||||||
|
</li>
|
||||||
|
|
||||||
|
<li v-if="timetable.skr">
|
||||||
|
<b class="text--primary">{{ $t('warnings.SKR') }}</b>
|
||||||
|
</li>
|
||||||
|
|
||||||
|
<li v-if="timetable.hasDangerousCargo">
|
||||||
|
<b class="text--primary">{{ $t('warnings.TN') }}</b>
|
||||||
|
</li>
|
||||||
|
|
||||||
|
<li v-if="timetable.hasExtraDeliveries">
|
||||||
|
<b class="text--primary">{{ $t('warnings.PN') }}</b>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<div class="dangers-notes" v-if="timetable.warningNotes">
|
||||||
|
<h4>{{ $t('warnings.header-title') }}</h4>
|
||||||
|
<p>
|
||||||
|
<i>{{ timetable.warningNotes }}</i>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Historia zmian w składzie -->
|
||||||
|
<div v-if="timetable.stockString || stockHistory.length != 0">
|
||||||
|
<div class="g-separator"></div>
|
||||||
|
|
||||||
|
<b>{{ $t('journal.stock-preview') }}:</b>
|
||||||
|
|
||||||
|
<div class="stock-specs" style="margin-top: 0.5em">
|
||||||
|
<span class="badge specs-badge" v-if="timetable.stockLength">
|
||||||
<span>{{ $t('journal.stock-length') }}</span>
|
<span>{{ $t('journal.stock-length') }}</span>
|
||||||
<span>
|
<span>
|
||||||
{{
|
{{
|
||||||
@@ -45,7 +89,7 @@
|
|||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="badge" v-if="timetable.stockMass">
|
<span class="badge specs-badge" v-if="timetable.stockMass">
|
||||||
<span>{{ $t('journal.stock-mass') }}</span>
|
<span>{{ $t('journal.stock-mass') }}</span>
|
||||||
<span>
|
<span>
|
||||||
{{
|
{{
|
||||||
@@ -59,34 +103,11 @@
|
|||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="stock-dangers" v-if="timetable.twr || timetable.skr">
|
<div class="stock-history">
|
||||||
<div class="g-separator"></div>
|
<button class="btn btn--action" @click="copyStockToClipboard()">
|
||||||
|
<i class="fa-regular fa-copy"></i> {{ $t('journal.stock-copy') }}
|
||||||
|
</button>
|
||||||
|
|
||||||
<b>{{ $t('journal.stock-dangers') }}:</b>
|
|
||||||
|
|
||||||
<ul>
|
|
||||||
<li v-if="timetable.twr">
|
|
||||||
<b class="text--primary">{{ $t('general.TWR') }} (TWR)</b>
|
|
||||||
<span v-if="timetable.warningNotes">
|
|
||||||
| <i>{{ timetable.warningNotes }}</i>
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
|
|
||||||
<li v-if="timetable.skr">
|
|
||||||
<b class="text--primary">{{ $t('general.SKR') }}</b>
|
|
||||||
<span v-if="timetable.warningNotes">
|
|
||||||
| Komentarze: <i>{{ timetable.warningNotes }}</i>
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Historia zmian w składzie -->
|
|
||||||
<div v-if="timetable.stockString || stockHistory.length != 0">
|
|
||||||
<div class="g-separator"></div>
|
|
||||||
<b>{{ $t('journal.stock-preview') }}:</b>
|
|
||||||
|
|
||||||
<div class="stock-history" v-if="stockHistory.length > 1">
|
|
||||||
<button
|
<button
|
||||||
v-for="(sh, i) in stockHistory"
|
v-for="(sh, i) in stockHistory"
|
||||||
:key="i"
|
:key="i"
|
||||||
@@ -119,6 +140,7 @@ import StockList from '../../Global/StockList.vue';
|
|||||||
import { API } from '../../../typings/api';
|
import { API } from '../../../typings/api';
|
||||||
import { RouteLocationRaw } from 'vue-router';
|
import { RouteLocationRaw } from 'vue-router';
|
||||||
import EntryStops from './EntryStops.vue';
|
import EntryStops from './EntryStops.vue';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: { StockList, EntryStops },
|
components: { StockList, EntryStops },
|
||||||
@@ -137,7 +159,8 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
currentHistoryIndex: 0
|
currentHistoryIndex: 0,
|
||||||
|
i18n: useI18n()
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
@@ -158,6 +181,7 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
driverRouteLocation(): RouteLocationRaw | null {
|
driverRouteLocation(): RouteLocationRaw | null {
|
||||||
if (this.timetable.terminated) return null;
|
if (this.timetable.terminated) return null;
|
||||||
return {
|
return {
|
||||||
@@ -176,15 +200,33 @@ export default defineComponent({
|
|||||||
|
|
||||||
toggleExtraInfo() {
|
toggleExtraInfo() {
|
||||||
this.$emit('toggleExtraInfo', this.timetable.id);
|
this.$emit('toggleExtraInfo', this.timetable.id);
|
||||||
|
},
|
||||||
|
|
||||||
|
copyStockToClipboard() {
|
||||||
|
const currentStockString =
|
||||||
|
this.stockHistory[this.currentHistoryIndex]?.stockString ?? this.timetable.stockString;
|
||||||
|
|
||||||
|
if (!currentStockString) {
|
||||||
|
alert(this.i18n.t('journal.stock-clipboard-failure'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
navigator.clipboard
|
||||||
|
.writeText(currentStockString)
|
||||||
|
.then(() => {
|
||||||
|
prompt(this.i18n.t('journal.stock-clipboard-success'), currentStockString);
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
alert(this.i18n.t('journal.stock-clipboard-failure'));
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/variables.scss';
|
@use '../../../styles/responsive';
|
||||||
@import '../../../styles/responsive.scss';
|
@use '../../../styles/badge';
|
||||||
@import '../../../styles/badge.scss';
|
|
||||||
|
|
||||||
.details-body {
|
.details-body {
|
||||||
margin-top: 0.5em;
|
margin-top: 0.5em;
|
||||||
@@ -207,22 +249,30 @@ export default defineComponent({
|
|||||||
margin-top: 1em;
|
margin-top: 1em;
|
||||||
|
|
||||||
button[data-checked='true'] {
|
button[data-checked='true'] {
|
||||||
color: $accentCol;
|
color: var(--clr-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.timetable-specs,
|
||||||
.stock-specs {
|
.stock-specs {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
.badge {
|
.specs-badge {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
|
|
||||||
|
span:first-child {
|
||||||
|
color: white;
|
||||||
|
background-color: #666;
|
||||||
|
border-radius: 0.25em 0 0 0.25em;
|
||||||
|
}
|
||||||
|
|
||||||
span:last-child {
|
span:last-child {
|
||||||
color: black;
|
color: black;
|
||||||
background-color: $accentCol;
|
background-color: var(--clr-primary);
|
||||||
}
|
border-radius: 0 0.25em 0.25em 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -234,10 +284,23 @@ hr {
|
|||||||
list-style: disc;
|
list-style: disc;
|
||||||
padding-left: 1em;
|
padding-left: 1em;
|
||||||
padding-top: 0.5em;
|
padding-top: 0.5em;
|
||||||
|
white-space: pre-wrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen() {
|
.dangers-notes {
|
||||||
.stock-specs {
|
margin-top: 0.5em;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
|
||||||
|
p {
|
||||||
|
margin-top: 0.25em;
|
||||||
|
max-height: 200px;
|
||||||
|
max-width: 500px;
|
||||||
|
overflow: auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@include responsive.smallScreen{
|
||||||
|
.timetable-specs {
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,21 +7,38 @@
|
|||||||
class="train-badge twr"
|
class="train-badge twr"
|
||||||
v-if="timetable.twr"
|
v-if="timetable.twr"
|
||||||
data-tooltip-type="BaseTooltip"
|
data-tooltip-type="BaseTooltip"
|
||||||
:data-tooltip-content="
|
:data-tooltip-content="$t('warnings.TWR')"
|
||||||
$t('general.TWR') + `${timetable.warningNotes ? ':\n' + timetable.warningNotes : ''}`
|
|
||||||
"
|
|
||||||
>
|
>
|
||||||
TWR
|
TWR
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span
|
<span
|
||||||
class="train-badge skr"
|
class="train-badge skr"
|
||||||
v-if="timetable.skr"
|
v-if="timetable.skr"
|
||||||
data-tooltip-type="BaseTooltip"
|
data-tooltip-type="BaseTooltip"
|
||||||
:data-tooltip-content="$t('general.SKR')"
|
:data-tooltip-content="$t('warnings.SKR')"
|
||||||
>
|
>
|
||||||
SKR
|
SKR
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
|
<span
|
||||||
|
class="train-badge tn"
|
||||||
|
v-if="timetable.hasDangerousCargo"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('warnings.TN')"
|
||||||
|
>
|
||||||
|
TN
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span
|
||||||
|
class="train-badge pn"
|
||||||
|
v-if="timetable.hasExtraDeliveries"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('warnings.PN')"
|
||||||
|
>
|
||||||
|
PN
|
||||||
|
</span>
|
||||||
|
|
||||||
<span>
|
<span>
|
||||||
<strong
|
<strong
|
||||||
data-tooltip-type="BaseTooltip"
|
data-tooltip-type="BaseTooltip"
|
||||||
@@ -113,8 +130,8 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/responsive';
|
@use '../../../styles/responsive';
|
||||||
@import '../../../styles/badge';
|
@use '../../../styles/badge';
|
||||||
|
|
||||||
.item-general {
|
.item-general {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -174,7 +191,7 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen{
|
||||||
.item-general {
|
.item-general {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
|||||||
@@ -21,7 +21,7 @@
|
|||||||
>
|
>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="text--grayed" v-if="timetable.currentSceneryName">
|
<span class="entry-location" v-if="timetable.currentSceneryName">
|
||||||
<b>
|
<b>
|
||||||
{{ $t(`journal.${timetable.terminated ? 'last-seen-at' : 'currently-at'}`) }}
|
{{ $t(`journal.${timetable.terminated ? 'last-seen-at' : 'currently-at'}`) }}
|
||||||
{{ timetable.currentSceneryName.replace(/.[a-zA-Z0-9]+.sc/, '') }}
|
{{ timetable.currentSceneryName.replace(/.[a-zA-Z0-9]+.sc/, '') }}
|
||||||
@@ -59,7 +59,7 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/responsive.scss';
|
@use '../../../styles/responsive';
|
||||||
|
|
||||||
.entry-status {
|
.entry-status {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -67,8 +67,13 @@ export default defineComponent({
|
|||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
|
|
||||||
@include smallScreen() {
|
@include responsive.smallScreen{
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.entry-location {
|
||||||
|
text-align: center;
|
||||||
|
color: #ccc;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -94,12 +94,13 @@ import { API } from '../../../typings/api';
|
|||||||
|
|
||||||
interface ITimetableStopDetails {
|
interface ITimetableStopDetails {
|
||||||
stopName: string;
|
stopName: string;
|
||||||
|
stopComments: string | null;
|
||||||
|
stopTime: number;
|
||||||
|
stopType: string;
|
||||||
arrivalTimestamp: number;
|
arrivalTimestamp: number;
|
||||||
scheduledArrivalTimestamp: number;
|
scheduledArrivalTimestamp: number;
|
||||||
departureTimestamp: number;
|
departureTimestamp: number;
|
||||||
scheduledDepartureTimestamp: number;
|
scheduledDepartureTimestamp: number;
|
||||||
stopTime: number;
|
|
||||||
stopType: string;
|
|
||||||
isConfirmed: boolean;
|
isConfirmed: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -164,15 +165,17 @@ export default defineComponent({
|
|||||||
|
|
||||||
const stopTime = Number(timetable.checkpointStopTypes.at(i)?.split(',')[0]) || 0;
|
const stopTime = Number(timetable.checkpointStopTypes.at(i)?.split(',')[0]) || 0;
|
||||||
const stopType = timetable.checkpointStopTypes.at(i)?.split(',').slice(1).join(',') || 'pt';
|
const stopType = timetable.checkpointStopTypes.at(i)?.split(',').slice(1).join(',') || 'pt';
|
||||||
|
const stopComments = timetable.checkpointComments.at(i) ?? null;
|
||||||
|
|
||||||
acc.push({
|
acc.push({
|
||||||
stopName,
|
stopName,
|
||||||
|
stopTime,
|
||||||
|
stopType,
|
||||||
|
stopComments,
|
||||||
arrivalTimestamp: this.dateStringToTimestamp(arrivalDate),
|
arrivalTimestamp: this.dateStringToTimestamp(arrivalDate),
|
||||||
scheduledArrivalTimestamp: this.dateStringToTimestamp(scheduledArrivalDate),
|
scheduledArrivalTimestamp: this.dateStringToTimestamp(scheduledArrivalDate),
|
||||||
departureTimestamp: this.dateStringToTimestamp(departureDate),
|
departureTimestamp: this.dateStringToTimestamp(departureDate),
|
||||||
scheduledDepartureTimestamp: this.dateStringToTimestamp(scheduledDepartureDate),
|
scheduledDepartureTimestamp: this.dateStringToTimestamp(scheduledDepartureDate),
|
||||||
stopTime,
|
|
||||||
stopType,
|
|
||||||
isConfirmed: i < timetable.confirmedStopsCount
|
isConfirmed: i < timetable.confirmedStopsCount
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -184,7 +187,7 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/badge.scss';
|
@use '../../../styles/badge';
|
||||||
|
|
||||||
.entry-stops {
|
.entry-stops {
|
||||||
word-wrap: break-word;
|
word-wrap: break-word;
|
||||||
@@ -218,6 +221,10 @@ export default defineComponent({
|
|||||||
.stop-name {
|
.stop-name {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: #ccc;
|
color: #ccc;
|
||||||
|
|
||||||
|
i {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.stop-date {
|
.stop-date {
|
||||||
|
|||||||
@@ -12,14 +12,6 @@
|
|||||||
<hr class="header-separator" />
|
<hr class="header-separator" />
|
||||||
|
|
||||||
<div class="info-stats">
|
<div class="info-stats">
|
||||||
<span class="badge stat-badge">
|
|
||||||
<span>{{ $t('journal.driver-stats.timetables') }}</span>
|
|
||||||
<span
|
|
||||||
>{{ store.driverStatsData._count.fulfilled }} /
|
|
||||||
{{ store.driverStatsData._count._all }}</span
|
|
||||||
>
|
|
||||||
</span>
|
|
||||||
|
|
||||||
<span class="badge stat-badge">
|
<span class="badge stat-badge">
|
||||||
<span>{{ $t('journal.driver-stats.longest-timetable') }}</span>
|
<span>{{ $t('journal.driver-stats.longest-timetable') }}</span>
|
||||||
<span> {{ store.driverStatsData._max.routeDistance.toFixed(2) }}km </span>
|
<span> {{ store.driverStatsData._max.routeDistance.toFixed(2) }}km </span>
|
||||||
@@ -29,12 +21,43 @@
|
|||||||
<span>{{ $t('journal.driver-stats.avg-timetable') }}</span>
|
<span>{{ $t('journal.driver-stats.avg-timetable') }}</span>
|
||||||
<span> {{ store.driverStatsData._avg.routeDistance.toFixed(2) }}km </span>
|
<span> {{ store.driverStatsData._avg.routeDistance.toFixed(2) }}km </span>
|
||||||
</span>
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr class="section-separator" />
|
||||||
|
|
||||||
|
<div class="info-stats">
|
||||||
|
<span class="badge stat-badge">
|
||||||
|
<span>{{ $t('journal.driver-stats.timetables') }}</span>
|
||||||
|
<span>
|
||||||
|
{{ store.driverStatsData._count.fulfilled }} /
|
||||||
|
{{ store.driverStatsData._count._all }}
|
||||||
|
|
||||||
|
<template v-if="store.driverStatsData._count._all > 0">
|
||||||
|
({{
|
||||||
|
(
|
||||||
|
(store.driverStatsData._count.fulfilled / store.driverStatsData._count._all) *
|
||||||
|
100
|
||||||
|
).toFixed(2)
|
||||||
|
}}%)
|
||||||
|
</template>
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
|
||||||
<span class="badge stat-badge">
|
<span class="badge stat-badge">
|
||||||
<span>{{ $t('journal.driver-stats.distance') }}</span>
|
<span>{{ $t('journal.driver-stats.distance') }}</span>
|
||||||
<span>
|
<span>
|
||||||
{{ store.driverStatsData._sum.currentDistance.toFixed(2) }} /
|
{{ store.driverStatsData._sum.currentDistance.toFixed(2) }} /
|
||||||
{{ store.driverStatsData._sum.routeDistance.toFixed(2) }}km
|
{{ store.driverStatsData._sum.routeDistance.toFixed(2) }}km
|
||||||
|
|
||||||
|
<template v-if="store.driverStatsData._sum.routeDistance > 0">
|
||||||
|
({{
|
||||||
|
(
|
||||||
|
(store.driverStatsData._sum.currentDistance /
|
||||||
|
store.driverStatsData._sum.routeDistance) *
|
||||||
|
100
|
||||||
|
).toFixed(2)
|
||||||
|
}}%)
|
||||||
|
</template>
|
||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
@@ -43,6 +66,16 @@
|
|||||||
<span>
|
<span>
|
||||||
{{ store.driverStatsData._sum.confirmedStopsCount }} /
|
{{ store.driverStatsData._sum.confirmedStopsCount }} /
|
||||||
{{ store.driverStatsData._sum.allStopsCount }}
|
{{ store.driverStatsData._sum.allStopsCount }}
|
||||||
|
|
||||||
|
<template v-if="store.driverStatsData._sum.allStopsCount > 0">
|
||||||
|
({{
|
||||||
|
(
|
||||||
|
(store.driverStatsData._sum.confirmedStopsCount /
|
||||||
|
store.driverStatsData._sum.allStopsCount) *
|
||||||
|
100
|
||||||
|
).toFixed(2)
|
||||||
|
}}%)
|
||||||
|
</template>
|
||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -68,5 +101,5 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/JournalStats.scss';
|
@use '../../../styles/journal-stats';
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -3,7 +3,6 @@
|
|||||||
<!-- General -->
|
<!-- General -->
|
||||||
<EntryGeneral :timetable="timetableEntry" />
|
<EntryGeneral :timetable="timetableEntry" />
|
||||||
|
|
||||||
<div @click="toggleExtraInfo" style="cursor: pointer">
|
|
||||||
<!-- Route -->
|
<!-- Route -->
|
||||||
<div class="entry-route">
|
<div class="entry-route">
|
||||||
<b>{{ timetableEntry.route.replace('|', ' - ') }}</b>
|
<b>{{ timetableEntry.route.replace('|', ' - ') }}</b>
|
||||||
@@ -11,6 +10,7 @@
|
|||||||
|
|
||||||
<hr />
|
<hr />
|
||||||
|
|
||||||
|
<div @click="toggleExtraInfo" style="cursor: pointer">
|
||||||
<!-- Status -->
|
<!-- Status -->
|
||||||
<EntryStatus :timetable="timetableEntry" />
|
<EntryStatus :timetable="timetableEntry" />
|
||||||
</div>
|
</div>
|
||||||
@@ -134,15 +134,20 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/responsive.scss';
|
@use '../../../styles/responsive';
|
||||||
|
|
||||||
.timetable-history-entry {
|
.timetable-history-entry {
|
||||||
background-color: #1a1a1a;
|
background-color: #1a1a1a;
|
||||||
padding: 1em;
|
padding: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
.entry-route {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
@include responsive.smallScreen{
|
||||||
.entry-route {
|
.entry-route {
|
||||||
|
justify-content: center;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -91,13 +91,10 @@ export default defineComponent({
|
|||||||
deep: true,
|
deep: true,
|
||||||
handler() {
|
handler() {
|
||||||
this.extraInfoIndexes.length = 0;
|
this.extraInfoIndexes.length = 0;
|
||||||
|
|
||||||
this.$nextTick(() => {
|
|
||||||
console.log(this.$el.querySelector('ul'));
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
toggleExtraInfo(id: number) {
|
toggleExtraInfo(id: number) {
|
||||||
const existingIdx = this.extraInfoIndexes.indexOf(id);
|
const existingIdx = this.extraInfoIndexes.indexOf(id);
|
||||||
@@ -110,10 +107,11 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/JournalSection.scss';
|
@use '../../../styles/animations';
|
||||||
@import '../../../styles/animations.scss';
|
@use '../../../styles/journal-section';
|
||||||
|
@use '../../../styles/responsive';
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen{
|
||||||
.journal_item-info {
|
.journal_item-info {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,19 @@
|
|||||||
export namespace Journal {
|
export namespace Journal {
|
||||||
export type DispatcherSearchKey = 'search-dispatcher' | 'search-station' | 'search-date';
|
export type DispatcherSearchKey =
|
||||||
|
| 'search-dispatcher'
|
||||||
|
| 'search-station'
|
||||||
|
| 'search-date-from'
|
||||||
|
| 'search-date-to';
|
||||||
|
|
||||||
export type TimetableSearchKey =
|
export type TimetableSearchKey =
|
||||||
| 'search-driver'
|
| 'search-driver'
|
||||||
| 'search-train'
|
| 'search-train'
|
||||||
| 'search-date'
|
| 'search-date-from'
|
||||||
| 'search-dispatcher'
|
| 'search-dispatcher'
|
||||||
| 'search-issuedFrom'
|
| 'search-issuedFrom'
|
||||||
| 'search-terminatingAt'
|
| 'search-terminatingAt'
|
||||||
| 'search-via';
|
| 'search-via'
|
||||||
|
| 'select-categoryCode';
|
||||||
|
|
||||||
export type TimetableSearchType = {
|
export type TimetableSearchType = {
|
||||||
[key in TimetableSearchKey]: string;
|
[key in TimetableSearchKey]: string;
|
||||||
@@ -19,7 +24,7 @@ export namespace Journal {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export type TimetableSorterKey = 'timetableId' | 'beginDate' | 'distance' | 'total-stops';
|
export type TimetableSorterKey = 'timetableId' | 'beginDate' | 'distance' | 'total-stops';
|
||||||
export type DispatcherSorterKey = 'timestampFrom' | 'duration';
|
export type DispatcherSorterKey = 'timestampFrom' | 'currentDuration';
|
||||||
|
|
||||||
export interface DispatcherSorter {
|
export interface DispatcherSorter {
|
||||||
id: DispatcherSorterKey;
|
id: DispatcherSorterKey;
|
||||||
@@ -39,6 +44,8 @@ export namespace Journal {
|
|||||||
ALL_SPECIALS = 'all-specials',
|
ALL_SPECIALS = 'all-specials',
|
||||||
TWR = 'twr',
|
TWR = 'twr',
|
||||||
SKR = 'skr',
|
SKR = 'skr',
|
||||||
|
PN = 'pn',
|
||||||
|
TN = 'tn',
|
||||||
TWR_SKR = 'twr-skr'
|
TWR_SKR = 'twr-skr'
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -78,4 +85,3 @@ export namespace Journal {
|
|||||||
isConfirmed: boolean;
|
isConfirmed: boolean;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -96,6 +96,7 @@ export default defineComponent({
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
historyList: [] as API.DispatcherHistory.Response,
|
historyList: [] as API.DispatcherHistory.Response,
|
||||||
|
lastStationName: '',
|
||||||
dataStatus: Status.Data.Loading,
|
dataStatus: Status.Data.Loading,
|
||||||
DataStatus: Status.Data,
|
DataStatus: Status.Data,
|
||||||
apiStore: useApiStore()
|
apiStore: useApiStore()
|
||||||
@@ -103,10 +104,10 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
async activated() {
|
async activated() {
|
||||||
// if (this.historyList.length == 0) {
|
this.historyList.length = 0;
|
||||||
|
|
||||||
const fetchedHistory = await this.fetchAPIData();
|
const fetchedHistory = await this.fetchAPIData();
|
||||||
if (fetchedHistory) this.historyList = fetchedHistory;
|
if (fetchedHistory) this.historyList = fetchedHistory;
|
||||||
// }
|
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
@@ -148,8 +149,8 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/sceneryViewTables.scss';
|
@use '../../styles/scenery-history-table';
|
||||||
|
|
||||||
.scenery-dispatchers-history {
|
.scenery-dispatchers-history {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@@ -194,7 +195,7 @@ export default defineComponent({
|
|||||||
color: springgreen;
|
color: springgreen;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen {
|
||||||
.journal-list > div {
|
.journal-list > div {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
|||||||
@@ -1,5 +1,13 @@
|
|||||||
<template>
|
<template>
|
||||||
<section class="info-header">
|
<section class="info-header">
|
||||||
|
<button
|
||||||
|
class="btn btn-return"
|
||||||
|
:title="$t('scenery.return-btn')"
|
||||||
|
@click="onReturnButtonClick"
|
||||||
|
>
|
||||||
|
<img src="/images/icon-back.svg" alt="return button" />
|
||||||
|
</button>
|
||||||
|
|
||||||
<a class="scenery-name" :href="station?.generalInfo?.url" target="_blank">
|
<a class="scenery-name" :href="station?.generalInfo?.url" target="_blank">
|
||||||
{{ stationName.replace(/_/g, ' ') }}
|
{{ stationName.replace(/_/g, ' ') }}
|
||||||
</a>
|
</a>
|
||||||
@@ -12,12 +20,21 @@
|
|||||||
</section>
|
</section>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts" setup>
|
||||||
import { PropType, defineComponent } from 'vue';
|
import { onMounted, PropType, ref } from 'vue';
|
||||||
import { ActiveScenery, Station } from '../../typings/common';
|
import { ActiveScenery, Station } from '../../typings/common';
|
||||||
|
import { useRoute, useRouter } from 'vue-router';
|
||||||
|
|
||||||
export default defineComponent({
|
const route = useRoute();
|
||||||
props: {
|
const router = useRouter();
|
||||||
|
|
||||||
|
const prevPath = ref('/');
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
prevPath.value = (route.meta['prevPath'] as string) ?? '/';
|
||||||
|
});
|
||||||
|
|
||||||
|
defineProps({
|
||||||
station: {
|
station: {
|
||||||
type: Object as PropType<Station>
|
type: Object as PropType<Station>
|
||||||
},
|
},
|
||||||
@@ -30,22 +47,37 @@ export default defineComponent({
|
|||||||
onlineScenery: {
|
onlineScenery: {
|
||||||
type: Object as PropType<ActiveScenery>
|
type: Object as PropType<ActiveScenery>
|
||||||
}
|
}
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function onReturnButtonClick() {
|
||||||
|
router.push(prevPath.value);
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/variables.scss';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/responsive.scss';
|
@use 'sass:color';
|
||||||
|
|
||||||
.info-header {
|
.btn-return {
|
||||||
margin-top: 1em;
|
$bgColor: #2b2b2b;
|
||||||
|
background-color: $bgColor;
|
||||||
|
margin-bottom: 0.5em;
|
||||||
|
|
||||||
|
img {
|
||||||
|
width: 2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: color.adjust($color: $bgColor, $lightness: 15%);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.scenery-name {
|
.scenery-name {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
font-size: 3em;
|
font-size: 3em;
|
||||||
|
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -59,4 +91,10 @@ export default defineComponent({
|
|||||||
color: #aaa;
|
color: #aaa;
|
||||||
font-size: 1.2em;
|
font-size: 1.2em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@include responsive.smallScreen {
|
||||||
|
.scenery-name {
|
||||||
|
font-size: 2.5em;
|
||||||
|
}
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,71 +1,12 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="scenery-info">
|
<div class="scenery-info">
|
||||||
<section>
|
<section>
|
||||||
<div class="scenery-info-general">
|
|
||||||
<SceneryInfoIcons :station="station" />
|
<SceneryInfoIcons :station="station" />
|
||||||
|
<SceneryInfoGeneral :station="station" />
|
||||||
<div class="scenery-general-list" v-if="station?.generalInfo">
|
|
||||||
<span>
|
|
||||||
<b>{{ $t('availability.title') }}:</b>
|
|
||||||
{{ $t(`availability.${station.generalInfo.availability}`) }}
|
|
||||||
|
|
||||||
<span v-if="station.generalInfo.reqLevel > -1">
|
|
||||||
-
|
|
||||||
{{
|
|
||||||
$t(
|
|
||||||
'scenery.req-level',
|
|
||||||
{ lvl: station.generalInfo.reqLevel },
|
|
||||||
station.generalInfo.reqLevel
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
|
|
||||||
<span>
|
|
||||||
• <b>{{ $t('controls.title') }}:</b>
|
|
||||||
{{ $t(`controls.${station.generalInfo.controlType}`) }}
|
|
||||||
</span>
|
|
||||||
|
|
||||||
<span>
|
|
||||||
• <b>{{ $t('signals.title') }}:</b>
|
|
||||||
{{ $t(`signals.${station.generalInfo.signalType}`) }}
|
|
||||||
</span>
|
|
||||||
|
|
||||||
<span v-if="station.generalInfo.lines">
|
|
||||||
• <b>{{ $t('scenery.lines-title') }}:</b> {{ station.generalInfo.lines }}
|
|
||||||
</span>
|
|
||||||
<span v-if="station.generalInfo.project">
|
|
||||||
• <b>{{ $t('scenery.project-title') }}: </b>
|
|
||||||
<a
|
|
||||||
style="color: salmon; text-decoration: underline; font-weight: bold"
|
|
||||||
:href="station.generalInfo.projectUrl"
|
|
||||||
target="_blank"
|
|
||||||
>
|
|
||||||
{{ station.generalInfo.project }}
|
|
||||||
</a>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<SceneryInfoRoutes v-if="station" :station="station" />
|
<SceneryInfoRoutes v-if="station" :station="station" />
|
||||||
|
<SceneryInfoAuthors :station="station" />
|
||||||
|
|
||||||
<div
|
<div style="margin: 1em 0; height: 2px; background-color: white"></div>
|
||||||
class="scenery-authors"
|
|
||||||
v-if="station?.generalInfo?.authors && station.generalInfo.authors.length > 0"
|
|
||||||
>
|
|
||||||
<b>
|
|
||||||
{{
|
|
||||||
$t(
|
|
||||||
'scenery.authors-title',
|
|
||||||
{ authors: station.generalInfo.authors.length },
|
|
||||||
station.generalInfo.authors.length
|
|
||||||
)
|
|
||||||
}}:
|
|
||||||
</b>
|
|
||||||
{{ station.generalInfo.authors.join(', ') }}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div style="margin: 2em 0; height: 2px; background-color: white"></div>
|
|
||||||
|
|
||||||
<!-- info dispatcher -->
|
<!-- info dispatcher -->
|
||||||
<SceneryInfoDispatcher :onlineScenery="onlineScenery" />
|
<SceneryInfoDispatcher :onlineScenery="onlineScenery" />
|
||||||
@@ -89,12 +30,17 @@ import SceneryInfoIcons from './SceneryInfo/SceneryInfoIcons.vue';
|
|||||||
import SceneryInfoUserList from './SceneryInfo/SceneryInfoUserList.vue';
|
import SceneryInfoUserList from './SceneryInfo/SceneryInfoUserList.vue';
|
||||||
import SceneryInfoSpawnList from './SceneryInfo/SceneryInfoSpawnList.vue';
|
import SceneryInfoSpawnList from './SceneryInfo/SceneryInfoSpawnList.vue';
|
||||||
import SceneryInfoRoutes from './SceneryInfo/SceneryInfoRoutes.vue';
|
import SceneryInfoRoutes from './SceneryInfo/SceneryInfoRoutes.vue';
|
||||||
|
import SceneryInfoGeneral from './SceneryInfo/SceneryInfoGeneral.vue';
|
||||||
|
import SceneryInfoAuthors from './SceneryInfo/SceneryInfoAuthors.vue';
|
||||||
|
|
||||||
import { ActiveScenery, Station } from '../../typings/common';
|
import { ActiveScenery, Station } from '../../typings/common';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: {
|
components: {
|
||||||
SceneryInfoDispatcher,
|
SceneryInfoDispatcher,
|
||||||
|
SceneryInfoGeneral,
|
||||||
SceneryInfoIcons,
|
SceneryInfoIcons,
|
||||||
|
SceneryInfoAuthors,
|
||||||
SceneryInfoUserList,
|
SceneryInfoUserList,
|
||||||
SceneryInfoSpawnList,
|
SceneryInfoSpawnList,
|
||||||
SceneryInfoRoutes
|
SceneryInfoRoutes
|
||||||
@@ -112,8 +58,8 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/badge.scss';
|
@use '../../styles/badge';
|
||||||
|
|
||||||
h3.section-header {
|
h3.section-header {
|
||||||
margin: 0.5em 0;
|
margin: 0.5em 0;
|
||||||
@@ -134,20 +80,6 @@ h3.section-header {
|
|||||||
margin-top: 1em;
|
margin-top: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.scenery-info-general {
|
|
||||||
margin-top: 1em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.scenery-general-list {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
|
|
||||||
span {
|
|
||||||
margin: 0 0.15em;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.scenery-topic a {
|
.scenery-topic a {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,30 @@
|
|||||||
|
<template>
|
||||||
|
<section
|
||||||
|
class="scenery-authors"
|
||||||
|
v-if="station?.generalInfo?.authors && station.generalInfo.authors.length > 0"
|
||||||
|
>
|
||||||
|
<b>
|
||||||
|
{{
|
||||||
|
$t(
|
||||||
|
'scenery.authors-title',
|
||||||
|
{ authors: station.generalInfo.authors.length },
|
||||||
|
station.generalInfo.authors.length
|
||||||
|
)
|
||||||
|
}}:
|
||||||
|
</b>
|
||||||
|
{{ station.generalInfo.authors.join(', ') }}
|
||||||
|
</section>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent, PropType } from 'vue';
|
||||||
|
import { Station } from '../../../typings/common';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
props: {
|
||||||
|
station: {
|
||||||
|
type: Object as PropType<Station>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
@@ -0,0 +1,92 @@
|
|||||||
|
<template>
|
||||||
|
<section class="info-general">
|
||||||
|
<div v-if="station?.generalInfo === undefined">
|
||||||
|
<b>{{ $t('scenery.no-data') }}</b>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div v-else>
|
||||||
|
<span>
|
||||||
|
<b>{{ $t('availability.title') }}:</b>
|
||||||
|
{{ $t(`availability.${station.generalInfo.availability}`) }}
|
||||||
|
|
||||||
|
<span v-if="station.generalInfo.reqLevel > -1">
|
||||||
|
-
|
||||||
|
{{
|
||||||
|
$t(
|
||||||
|
'scenery.req-level',
|
||||||
|
{ lvl: station.generalInfo.reqLevel },
|
||||||
|
station.generalInfo.reqLevel
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span>
|
||||||
|
• <b>{{ $t('controls.title') }}:</b>
|
||||||
|
{{ $t(`controls.${station.generalInfo.controlType}`) }}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span>
|
||||||
|
• <b>{{ $t('signals.title') }}:</b>
|
||||||
|
{{ $t(`signals.${station.generalInfo.signalType}`) }}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span v-if="station.generalInfo.lines">
|
||||||
|
• <b>{{ $t('scenery.lines-title') }}:</b> {{ station.generalInfo.lines }}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span v-if="station.generalInfo.project">
|
||||||
|
• <b>{{ $t('scenery.project-title') }}: </b>
|
||||||
|
<a
|
||||||
|
style="color: salmon; text-decoration: underline; font-weight: bold"
|
||||||
|
:href="station.generalInfo.projectUrl"
|
||||||
|
target="_blank"
|
||||||
|
>
|
||||||
|
{{ station.generalInfo.project }}
|
||||||
|
</a>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span v-if="additionalTools.length != 0">
|
||||||
|
• <b>{{ $t('scenery.additional-tools-title') }}: </b>
|
||||||
|
{{ additionalTools.join(', ') }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent, PropType } from 'vue';
|
||||||
|
import { Station } from '../../../typings/common';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
props: {
|
||||||
|
station: {
|
||||||
|
type: Object as PropType<Station>
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
additionalTools() {
|
||||||
|
if (this.$props.station?.generalInfo === undefined) return [];
|
||||||
|
let tools = [];
|
||||||
|
|
||||||
|
if (this.$props.station.generalInfo.SUP) tools.push('SUP');
|
||||||
|
if (this.$props.station.generalInfo.ASDEK) tools.push('ASDEK');
|
||||||
|
|
||||||
|
return tools;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.info-general {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
|
||||||
|
div {
|
||||||
|
margin: 0 0.15em;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -17,25 +17,6 @@
|
|||||||
{{ station?.generalInfo.reqLevel >= 2 ? station?.generalInfo.reqLevel : 'L' }}
|
{{ station?.generalInfo.reqLevel >= 2 ? station?.generalInfo.reqLevel : 'L' }}
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span
|
|
||||||
v-if="station?.generalInfo"
|
|
||||||
class="scenery-icon icon-info"
|
|
||||||
:class="station?.generalInfo.controlType.replace('+', '-')"
|
|
||||||
:title="
|
|
||||||
$t('sceneries.info.control-type') + $t(`controls.${station?.generalInfo.controlType}`)
|
|
||||||
"
|
|
||||||
>
|
|
||||||
{{ $t(`controls.abbrevs.${station.generalInfo.controlType}`) }}
|
|
||||||
</span>
|
|
||||||
|
|
||||||
<img
|
|
||||||
v-if="station?.generalInfo?.signalType"
|
|
||||||
class="icon-info"
|
|
||||||
:src="`/images/icon-${station.generalInfo.signalType}.svg`"
|
|
||||||
:alt="station.generalInfo.signalType"
|
|
||||||
:title="$t('sceneries.info.signals-type') + $t(`signals.${station.generalInfo.signalType}`)"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<img
|
<img
|
||||||
v-if="station?.generalInfo?.availability == 'nonPublic'"
|
v-if="station?.generalInfo?.availability == 'nonPublic'"
|
||||||
class="icon-info"
|
class="icon-info"
|
||||||
@@ -60,6 +41,33 @@
|
|||||||
:title="$t('sceneries.info.abandoned')"
|
:title="$t('sceneries.info.abandoned')"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<span
|
||||||
|
v-if="station?.generalInfo"
|
||||||
|
class="scenery-icon icon-info"
|
||||||
|
:class="station?.generalInfo.controlType.replace('+', '-')"
|
||||||
|
:title="
|
||||||
|
$t('sceneries.info.control-type') + $t(`controls.${station?.generalInfo.controlType}`)
|
||||||
|
"
|
||||||
|
>
|
||||||
|
{{ $t(`controls.abbrevs.${station.generalInfo.controlType}`) }}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<img
|
||||||
|
v-if="station?.generalInfo?.signalType"
|
||||||
|
class="icon-info"
|
||||||
|
:src="`/images/icon-${station.generalInfo.signalType}.svg`"
|
||||||
|
:alt="station.generalInfo.signalType"
|
||||||
|
:title="$t('sceneries.info.signals-type') + $t(`signals.${station.generalInfo.signalType}`)"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<img
|
||||||
|
v-if="station?.generalInfo?.lines"
|
||||||
|
class="icon-info"
|
||||||
|
src="/images/icon-real.svg"
|
||||||
|
alt="real scenery"
|
||||||
|
:title="`${$t('sceneries.info.real')} ${station.generalInfo.lines}`"
|
||||||
|
/>
|
||||||
|
|
||||||
<img
|
<img
|
||||||
v-if="station?.generalInfo?.SUP"
|
v-if="station?.generalInfo?.SUP"
|
||||||
class="icon-info"
|
class="icon-info"
|
||||||
@@ -75,14 +83,6 @@
|
|||||||
alt="dSAT ASDEK"
|
alt="dSAT ASDEK"
|
||||||
:title="$t('sceneries.info.ASDEK')"
|
:title="$t('sceneries.info.ASDEK')"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<img
|
|
||||||
v-if="station?.generalInfo?.lines"
|
|
||||||
class="icon-info"
|
|
||||||
src="/images/icon-real.svg"
|
|
||||||
alt="real scenery"
|
|
||||||
:title="`${$t('sceneries.info.real')} ${station.generalInfo.lines}`"
|
|
||||||
/>
|
|
||||||
</section>
|
</section>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -102,7 +102,7 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/icons.scss';
|
@use '../../../styles/icons';
|
||||||
|
|
||||||
.info-icons {
|
.info-icons {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -118,6 +118,7 @@ export default defineComponent({
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
|
|
||||||
width: 3em;
|
width: 3em;
|
||||||
|
height: 3em;
|
||||||
margin: 0.25em;
|
margin: 0.25em;
|
||||||
|
|
||||||
border: 2px solid #4e4e4e;
|
border: 2px solid #4e4e4e;
|
||||||
|
|||||||
@@ -1,23 +1,26 @@
|
|||||||
<template>
|
<template>
|
||||||
<section class="info-routes" v-if="station.generalInfo">
|
<section class="info-routes" v-if="station.generalInfo">
|
||||||
<div class="routes one-way" v-if="oneWayRoutes.length > 0">
|
<div class="routes one-way" v-if="oneWayRoutes.length > 0">
|
||||||
|
<button
|
||||||
|
class="routes-btn"
|
||||||
|
@click="toggleRoutesVisibility('single')"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="`${showInternalSingleRoutes ? $t('scenery.btn-hide-internal-routes') : $t('scenery.btn-show-internal-routes')}`"
|
||||||
|
>
|
||||||
<b>{{ $t('scenery.one-way-routes') }}</b>
|
<b>{{ $t('scenery.one-way-routes') }}</b>
|
||||||
|
<i class="fa-solid" :class="`${showInternalSingleRoutes ? 'fa-eye' : 'fa-eye-slash'}`"></i>
|
||||||
|
</button>
|
||||||
|
|
||||||
<ul class="routes-list">
|
<ul class="routes-list">
|
||||||
<li
|
<li v-for="route in oneWayRoutes" :key="route.routeName">
|
||||||
v-for="route in oneWayRoutes"
|
|
||||||
:key="route.routeName"
|
|
||||||
@click="setActiveShowLength(route.routeName)"
|
|
||||||
>
|
|
||||||
<span :class="{ 'no-catenary': !route.isElectric, internal: route.isInternal }">
|
<span :class="{ 'no-catenary': !route.isElectric, internal: route.isInternal }">
|
||||||
{{ route.routeName }}</span
|
{{ route.routeName }}</span
|
||||||
>
|
>
|
||||||
<span v-if="route.routeSpeed" class="speed">
|
<span v-if="route.routeSpeed" class="speed">
|
||||||
{{
|
{{ route.routeSpeed }}
|
||||||
activeShowLength.includes(route.routeName)
|
</span>
|
||||||
? route.routeLength + 'm'
|
<span v-if="route.routeLength" class="length">
|
||||||
: route.routeSpeed
|
{{ (route.routeLength / 1000).toFixed(1) + 'km' }}
|
||||||
}}
|
|
||||||
</span>
|
</span>
|
||||||
<span v-if="route.isRouteSBL" class="sbl">SBL</span>
|
<span v-if="route.isRouteSBL" class="sbl">SBL</span>
|
||||||
</li>
|
</li>
|
||||||
@@ -25,23 +28,29 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="routes two-way" v-if="twoWayRoutes.length > 0">
|
<div class="routes two-way" v-if="twoWayRoutes.length > 0">
|
||||||
|
<button
|
||||||
|
class="routes-btn"
|
||||||
|
@click="toggleRoutesVisibility('double')"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="`${showInternalDoubleRoutes ? $t('scenery.btn-hide-internal-routes') : $t('scenery.btn-show-internal-routes')}`"
|
||||||
|
>
|
||||||
<b>{{ $t('scenery.two-way-routes') }}</b>
|
<b>{{ $t('scenery.two-way-routes') }}</b>
|
||||||
|
<i class="fa-solid" :class="`${showInternalDoubleRoutes ? 'fa-eye' : 'fa-eye-slash'}`"></i>
|
||||||
|
</button>
|
||||||
|
|
||||||
<ul class="routes-list">
|
<ul class="routes-list">
|
||||||
<li
|
<li v-for="route in twoWayRoutes" :key="route.routeName">
|
||||||
v-for="route in twoWayRoutes"
|
<span :class="{ 'no-catenary': !route.isElectric, internal: route.isInternal }">
|
||||||
:key="route.routeName"
|
{{ route.routeName }}
|
||||||
@click="setActiveShowLength(route.routeName)"
|
</span>
|
||||||
>
|
|
||||||
<span :class="{ 'no-catenary': !route.isElectric, internal: route.isInternal }">{{
|
|
||||||
route.routeName
|
|
||||||
}}</span>
|
|
||||||
<span v-if="route.routeSpeed" class="speed">
|
<span v-if="route.routeSpeed" class="speed">
|
||||||
{{
|
<span>{{ route.routeSpeed }}</span>
|
||||||
activeShowLength.includes(route.routeName)
|
<span v-if="route.routeSpeedExit && route.routeSpeedExit != route.routeSpeed">
|
||||||
? route.routeLength + 'm'
|
| {{ route.routeSpeedExit }}
|
||||||
: route.routeSpeed
|
</span>
|
||||||
}}
|
</span>
|
||||||
|
<span v-if="route.routeLength" class="length">
|
||||||
|
{{ (route.routeLength / 1000).toFixed(1) + 'km' }}
|
||||||
</span>
|
</span>
|
||||||
<span v-if="route.isRouteSBL" class="sbl">SBL</span>
|
<span v-if="route.isRouteSBL" class="sbl">SBL</span>
|
||||||
</li>
|
</li>
|
||||||
@@ -53,6 +62,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { PropType, defineComponent } from 'vue';
|
import { PropType, defineComponent } from 'vue';
|
||||||
import { Station } from '../../../typings/common';
|
import { Station } from '../../../typings/common';
|
||||||
|
import StorageManager from '../../../managers/storageManager';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
props: {
|
props: {
|
||||||
@@ -62,27 +72,50 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
data() {
|
||||||
setActiveShowLength(name: string) {
|
return {
|
||||||
if (this.activeShowLength.includes(name))
|
showInternalSingleRoutes: false,
|
||||||
this.activeShowLength.splice(this.activeShowLength.indexOf(name), 1);
|
showInternalDoubleRoutes: false
|
||||||
else this.activeShowLength.push(name);
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
mounted() {
|
||||||
|
if (StorageManager.getBooleanValue('showInternalDoubleRoutes')) {
|
||||||
|
this.showInternalDoubleRoutes = StorageManager.getBooleanValue('showInternalDoubleRoutes');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (StorageManager.getBooleanValue('showInternalSingleRoutes')) {
|
||||||
|
this.showInternalSingleRoutes = StorageManager.getBooleanValue('showInternalSingleRoutes');
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
data() {
|
methods: {
|
||||||
return {
|
toggleRoutesVisibility(type: 'single' | 'double') {
|
||||||
activeShowLength: [] as string[]
|
if (type == 'double') {
|
||||||
};
|
this.showInternalDoubleRoutes = !this.showInternalDoubleRoutes;
|
||||||
|
StorageManager.setBooleanValue('showInternalDoubleRoutes', this.showInternalDoubleRoutes);
|
||||||
|
} else {
|
||||||
|
this.showInternalSingleRoutes = !this.showInternalSingleRoutes;
|
||||||
|
StorageManager.setBooleanValue('showInternalSingleRoutes', this.showInternalSingleRoutes);
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
oneWayRoutes() {
|
oneWayRoutes() {
|
||||||
return this.station.generalInfo?.routes.single ?? [];
|
return (
|
||||||
|
this.station.generalInfo?.routes.single
|
||||||
|
.filter((r) => !r.isInternal || r.isInternal == this.showInternalSingleRoutes)
|
||||||
|
.sort((r1, r2) => r1.routeName.localeCompare(r2.routeName)) ?? []
|
||||||
|
);
|
||||||
},
|
},
|
||||||
|
|
||||||
twoWayRoutes() {
|
twoWayRoutes() {
|
||||||
return this.station.generalInfo?.routes.double ?? [];
|
return (
|
||||||
|
this.station.generalInfo?.routes.double
|
||||||
|
.filter((r) => !r.isInternal || r.isInternal == this.showInternalDoubleRoutes)
|
||||||
|
.sort((r1, r2) => r1.routeName.localeCompare(r2.routeName)) ?? []
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -90,22 +123,29 @@ export default defineComponent({
|
|||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.info-routes {
|
.info-routes {
|
||||||
display: flex;
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
flex-wrap: wrap;
|
flex-direction: column;
|
||||||
|
|
||||||
margin: 1em 0;
|
margin: 1em 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.routes {
|
.routes {
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
|
|
||||||
padding: 0.25em;
|
padding: 0.25em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.routes > button.routes-btn {
|
||||||
|
margin: 0 auto;
|
||||||
|
display: inline-block;
|
||||||
|
|
||||||
|
i {
|
||||||
|
margin-left: 0.5em;
|
||||||
|
width: 1.25em;
|
||||||
|
height: 1.25em;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ul.routes-list {
|
ul.routes-list {
|
||||||
margin: 0.45em 0.25em;
|
margin: 0.45em 0.25em;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -120,8 +160,8 @@ ul.routes-list {
|
|||||||
-moz-user-select: none;
|
-moz-user-select: none;
|
||||||
-webkit-user-select: none;
|
-webkit-user-select: none;
|
||||||
|
|
||||||
span {
|
& > span {
|
||||||
padding: 0.2em 0.25em;
|
padding: 0.2em;
|
||||||
background-color: #007599;
|
background-color: #007599;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
|
|
||||||
@@ -138,6 +178,10 @@ ul.routes-list {
|
|||||||
color: #cfcfcf;
|
color: #cfcfcf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.length {
|
||||||
|
background-color: #303030;
|
||||||
|
color: #cfcfcf;
|
||||||
|
}
|
||||||
&.sbl {
|
&.sbl {
|
||||||
color: var(--clr-primary);
|
color: var(--clr-primary);
|
||||||
background-color: #404040;
|
background-color: #404040;
|
||||||
|
|||||||
@@ -53,8 +53,6 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../../styles/variables.scss';
|
|
||||||
|
|
||||||
ul {
|
ul {
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,9 +18,33 @@
|
|||||||
:key="train.id"
|
:key="train.id"
|
||||||
:data-status="status"
|
:data-status="status"
|
||||||
>
|
>
|
||||||
<router-link :to="train.driverRouteLocation" class="a-block">
|
<router-link
|
||||||
<span class="user_train">{{ train.trainNo }}</span>
|
:to="train.driverRouteLocation"
|
||||||
<span class="user_name">{{ train.driverName }}</span>
|
data-tooltip-type="TrainInfoTooltip"
|
||||||
|
:data-tooltip-content="train.id"
|
||||||
|
>
|
||||||
|
<span class="user_train"> {{ train.trainNo }}</span>
|
||||||
|
<span class="user_name">
|
||||||
|
{{ train.driverName }}
|
||||||
|
<i
|
||||||
|
v-if="
|
||||||
|
train.timetableData != undefined &&
|
||||||
|
(train.lastSeen <= Date.now() - 60000 || !train.online)
|
||||||
|
"
|
||||||
|
class="fa-solid fa-user-slash"
|
||||||
|
style="color: lightcoral"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('app.tooltip-driver-offline')"
|
||||||
|
></i>
|
||||||
|
|
||||||
|
<i
|
||||||
|
v-if="train.currentStationName.indexOf('.sc') != -1"
|
||||||
|
class="fa-solid fa-ban"
|
||||||
|
style="color: lightcoral"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('app.tooltip-scenery-offline')"
|
||||||
|
></i>
|
||||||
|
</span>
|
||||||
</router-link>
|
</router-link>
|
||||||
</li>
|
</li>
|
||||||
</transition-group>
|
</transition-group>
|
||||||
@@ -63,11 +87,17 @@ export default defineComponent({
|
|||||||
const stop = train.timetableData?.followingStops.find(
|
const stop = train.timetableData?.followingStops.find(
|
||||||
(stop) =>
|
(stop) =>
|
||||||
stop.stopNameRAW.toLowerCase() == name.toLowerCase() ||
|
stop.stopNameRAW.toLowerCase() == name.toLowerCase() ||
|
||||||
this.station?.generalInfo?.checkpoints.includes(stop.stopNameRAW)
|
this.station?.generalInfo?.checkpoints.includes(stop.stopNameRAW) ||
|
||||||
|
this.onlineScenery?.missingCheckpoints.includes(stop.stopNameRAW)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const sceneryName =
|
||||||
|
train.currentStationName.indexOf('.sc') != -1
|
||||||
|
? train.currentStationName.split(' ').slice(0, -1).join(' ')
|
||||||
|
: train.currentStationName;
|
||||||
|
|
||||||
const status = stop
|
const status = stop
|
||||||
? getTrainStopStatus(stop, train.currentStationName, this.onlineScenery!.name)
|
? getTrainStopStatus(stop, sceneryName, this.onlineScenery!.name)
|
||||||
: 'no-timetable';
|
: 'no-timetable';
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -13,12 +13,31 @@
|
|||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="header_links" v-if="station">
|
<span class="header_links" v-if="station && onlineScenery">
|
||||||
<a :href="pragotronHref" target="_blank" :title="$t('scenery.pragotron-link')">
|
<a
|
||||||
|
:href="generatorHref"
|
||||||
|
target="_blank"
|
||||||
|
data-tooltip-type="HtmlTooltip"
|
||||||
|
:data-tooltip-content="`<b>${$t('scenery.gnr-link')}</b>`"
|
||||||
|
>
|
||||||
|
<img src="/images/icon-gnr.svg" alt="GeneraTOR app icon" />
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<a
|
||||||
|
:href="pragotronHref"
|
||||||
|
target="_blank"
|
||||||
|
data-tooltip-type="HtmlTooltip"
|
||||||
|
:data-tooltip-content="`<b>${$t('scenery.pragotron-link')}</b>`"
|
||||||
|
>
|
||||||
<img src="/images/icon-pragotron.svg" alt="icon-pragotron" />
|
<img src="/images/icon-pragotron.svg" alt="icon-pragotron" />
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
<a :href="tabliceZbiorczeHref" target="_blank" :title="$t('scenery.tablice-link')">
|
<a
|
||||||
|
:href="tabliceZbiorczeHref"
|
||||||
|
target="_blank"
|
||||||
|
data-tooltip-type="HtmlTooltip"
|
||||||
|
:data-tooltip-content="`<b>${$t('scenery.tablice-link')}</b>`"
|
||||||
|
>
|
||||||
<img src="/images/icon-tablice.ico" alt="icon-tablice" />
|
<img src="/images/icon-tablice.ico" alt="icon-tablice" />
|
||||||
</a>
|
</a>
|
||||||
</span>
|
</span>
|
||||||
@@ -35,6 +54,18 @@
|
|||||||
>
|
>
|
||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="timetable-checkpoints" v-else-if="onlineScenery">
|
||||||
|
<template v-for="(ch, i) in onlineScenery.missingCheckpoints" :key="i">
|
||||||
|
<template v-if="i > 0">•</template>
|
||||||
|
<router-link
|
||||||
|
class="checkpoint-item"
|
||||||
|
:class="{ current: chosenCheckpoint === ch }"
|
||||||
|
:to="`/scenery?station=${onlineScenery.name}&checkpoint=${ch}`"
|
||||||
|
>{{ ch }}</router-link
|
||||||
|
>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="timetable-list">
|
<div class="timetable-list">
|
||||||
@@ -64,7 +95,7 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<router-link
|
<router-link
|
||||||
class="timetable-item a-block"
|
class="timetable-item"
|
||||||
v-else
|
v-else
|
||||||
v-for="(row, i) in sceneryTimetables"
|
v-for="(row, i) in sceneryTimetables"
|
||||||
:key="row.train.id + i"
|
:key="row.train.id + i"
|
||||||
@@ -74,19 +105,59 @@
|
|||||||
<span class="timetable-general">
|
<span class="timetable-general">
|
||||||
<span class="general-info">
|
<span class="general-info">
|
||||||
<div class="info-train">
|
<div class="info-train">
|
||||||
<b
|
<!-- Cargo warnings & details badges -->
|
||||||
|
<span
|
||||||
|
class="train-badge twr"
|
||||||
|
v-if="row.train.timetableData!.twr"
|
||||||
data-tooltip-type="BaseTooltip"
|
data-tooltip-type="BaseTooltip"
|
||||||
:data-tooltip-content="getCategoryExplanation(row.train.timetableData!.category)"
|
:data-tooltip-content="$t('warnings.TWR')"
|
||||||
class="text--primary tooltip-help"
|
|
||||||
>
|
>
|
||||||
|
TWR
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span
|
||||||
|
class="train-badge tn"
|
||||||
|
v-if="row.train.timetableData!.hasDangerousCargo"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('warnings.TN')"
|
||||||
|
>
|
||||||
|
TN
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span
|
||||||
|
class="train-badge pn"
|
||||||
|
v-if="row.train.timetableData!.hasExtraDeliveries"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('warnings.PN')"
|
||||||
|
>
|
||||||
|
PN
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<!-- Train info -->
|
||||||
|
<span
|
||||||
|
data-tooltip-type="TrainInfoTooltip"
|
||||||
|
:data-tooltip-content="row.train.id"
|
||||||
|
class="tooltip-help"
|
||||||
|
>
|
||||||
|
<b class="text--primary">
|
||||||
{{ row.train.timetableData!.category }}
|
{{ row.train.timetableData!.category }}
|
||||||
</b>
|
</b>
|
||||||
<span> </span>
|
|
||||||
<b>{{ row.train.trainNo }}</b>
|
<b> {{ row.train.trainNo }}</b>
|
||||||
<span> • </span>
|
•
|
||||||
<span>{{ row.train.driverName }}</span>
|
{{ row.train.driverName }}
|
||||||
|
|
||||||
|
<i
|
||||||
|
class="fa-solid fa-user-slash"
|
||||||
|
style="color: salmon"
|
||||||
|
v-if="!row.train.online && row.train.lastSeen <= Date.now() - 60000"
|
||||||
|
></i>
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<!-- Train stop comments -->
|
||||||
<span
|
<span
|
||||||
v-if="row.checkpointStop.comments"
|
v-if="row.checkpointStop.comments"
|
||||||
|
class="stop-comments-icon"
|
||||||
data-tooltip-type="BaseTooltip"
|
data-tooltip-type="BaseTooltip"
|
||||||
:data-tooltip-content="row.checkpointStop.comments"
|
:data-tooltip-content="row.checkpointStop.comments"
|
||||||
>
|
>
|
||||||
@@ -130,7 +201,7 @@
|
|||||||
|
|
||||||
<span class="schedule-stop">
|
<span class="schedule-stop">
|
||||||
<span class="stop-connection">
|
<span class="stop-connection">
|
||||||
{{ row.arrivingLine }}
|
{{ row.currentElement.arrivalRouteExt }}
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="stop-time">
|
<span class="stop-time">
|
||||||
@@ -139,7 +210,7 @@
|
|||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span class="stop-connection">
|
<span class="stop-connection">
|
||||||
{{ row.departureLine }}
|
{{ row.currentElement.departureRouteExt }}
|
||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
@@ -186,7 +257,7 @@ import { useMainStore } from '../../store/mainStore';
|
|||||||
import { useApiStore } from '../../store/apiStore';
|
import { useApiStore } from '../../store/apiStore';
|
||||||
import ScheduledTrainStatus from './ScheduledTrainStatus.vue';
|
import ScheduledTrainStatus from './ScheduledTrainStatus.vue';
|
||||||
import { SceneryTimetableRow } from './typings';
|
import { SceneryTimetableRow } from './typings';
|
||||||
import { ActiveScenery, Station } from '../../typings/common';
|
import { ActiveScenery, Station, TooltipTrainInfo, Train } from '../../typings/common';
|
||||||
import { getTrainStopStatus, stopStatusPriority } from './utils';
|
import { getTrainStopStatus, stopStatusPriority } from './utils';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
@@ -228,6 +299,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
const chosenCheckpoint = ref(
|
const chosenCheckpoint = ref(
|
||||||
props.station?.generalInfo?.checkpoints[0] ??
|
props.station?.generalInfo?.checkpoints[0] ??
|
||||||
|
props.onlineScenery?.missingCheckpoints[0] ??
|
||||||
props.station?.name ??
|
props.station?.name ??
|
||||||
route.query['station']?.toString() ??
|
route.query['station']?.toString() ??
|
||||||
''
|
''
|
||||||
@@ -256,6 +328,10 @@ export default defineComponent({
|
|||||||
return url;
|
return url;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
generatorHref() {
|
||||||
|
return `https://generator-td2.web.app/?sceneryId=${this.onlineScenery!.name}|${this.onlineScenery!.region}`;
|
||||||
|
},
|
||||||
|
|
||||||
sceneryTimetables(): SceneryTimetableRow[] {
|
sceneryTimetables(): SceneryTimetableRow[] {
|
||||||
if (!this.onlineScenery) return [];
|
if (!this.onlineScenery) return [];
|
||||||
|
|
||||||
@@ -279,12 +355,9 @@ export default defineComponent({
|
|||||||
return {
|
return {
|
||||||
checkpointStop: ct.checkpointStop,
|
checkpointStop: ct.checkpointStop,
|
||||||
train: ct.train,
|
train: ct.train,
|
||||||
prevDepartureLine: ct.previousSceneryElement?.departureRouteExt ?? null,
|
prevElement: ct.previousSceneryElement,
|
||||||
nextArrivalLine: ct.nextSceneryElement?.arrivalRouteExt ?? null,
|
nextElement: ct.nextSceneryElement,
|
||||||
departureLine: ct.timetablePathElement.departureRouteExt ?? null,
|
currentElement: ct.timetablePathElement,
|
||||||
arrivingLine: ct.timetablePathElement.arrivalRouteExt ?? null,
|
|
||||||
prevStationName: ct.previousSceneryElement?.stationName ?? null,
|
|
||||||
nextStationName: ct.nextSceneryElement?.stationName ?? null,
|
|
||||||
status: trainStopStatus
|
status: trainStopStatus
|
||||||
};
|
};
|
||||||
})
|
})
|
||||||
@@ -305,21 +378,30 @@ export default defineComponent({
|
|||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
loadSelectedOption() {
|
loadSelectedOption() {
|
||||||
if (!this.station) return;
|
|
||||||
|
|
||||||
if (!this.station.generalInfo) {
|
|
||||||
this.chosenCheckpoint = this.station.name;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const queryCheckpoint = this.$route.query['checkpoint']?.toString();
|
const queryCheckpoint = this.$route.query['checkpoint']?.toString();
|
||||||
|
|
||||||
|
let checkpointsListRef: string[] | null = null;
|
||||||
|
let sceneryName = '';
|
||||||
|
|
||||||
|
if (this.station && this.station.generalInfo) {
|
||||||
|
checkpointsListRef = this.station.generalInfo.checkpoints;
|
||||||
|
sceneryName = this.station.name;
|
||||||
|
} else if (this.onlineScenery) {
|
||||||
|
checkpointsListRef = this.onlineScenery.missingCheckpoints;
|
||||||
|
sceneryName = this.onlineScenery.name;
|
||||||
|
} else if (this.station) {
|
||||||
|
this.chosenCheckpoint = this.station.name;
|
||||||
|
sceneryName = this.station.name;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (checkpointsListRef) {
|
||||||
this.chosenCheckpoint =
|
this.chosenCheckpoint =
|
||||||
this.station.generalInfo.checkpoints.find(
|
checkpointsListRef.find(
|
||||||
(ch) => ch.toLocaleLowerCase() === queryCheckpoint?.toLocaleLowerCase()
|
(ch) => ch.toLocaleLowerCase() === queryCheckpoint?.toLocaleLowerCase()
|
||||||
) ??
|
) ??
|
||||||
this.station.generalInfo.checkpoints[0] ??
|
checkpointsListRef[0] ??
|
||||||
this.station.name;
|
sceneryName;
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
setCheckpoint(cp: string) {
|
setCheckpoint(cp: string) {
|
||||||
@@ -330,9 +412,9 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/variables.scss';
|
@use '../../styles/animations';
|
||||||
@import '../../styles/animations.scss';
|
@use '../../styles/badge';
|
||||||
|
|
||||||
.scenery-timetable {
|
.scenery-timetable {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@@ -367,7 +449,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
.header_links {
|
.header_links {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 0.5em;
|
gap: 0.25em;
|
||||||
margin-left: 0.5em;
|
margin-left: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -439,7 +521,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
&.current {
|
&.current {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: $accentCol;
|
color: var(--clr-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -449,21 +531,32 @@ export default defineComponent({
|
|||||||
|
|
||||||
.general-info {
|
.general-info {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
|
||||||
.info-number {
|
.info-train {
|
||||||
color: $accentCol;
|
display: flex;
|
||||||
}
|
flex-wrap: wrap;
|
||||||
|
gap: 0.25em;
|
||||||
|
}
|
||||||
|
|
||||||
.info-route {
|
.info-train > .train-badge {
|
||||||
|
font-size: 0.85em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-number {
|
||||||
|
color: var(--clr-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.info-route {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
margin-top: 0.25em;
|
||||||
|
}
|
||||||
|
|
||||||
img {
|
.stop-comments-icon > img {
|
||||||
height: 0.9em;
|
width: 1.3em;
|
||||||
vertical-align: middle;
|
vertical-align: top;
|
||||||
margin: 0 0.25em;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.schedule {
|
.schedule {
|
||||||
@@ -488,7 +581,7 @@ export default defineComponent({
|
|||||||
align-self: center;
|
align-self: center;
|
||||||
font-size: 0.9em;
|
font-size: 0.9em;
|
||||||
|
|
||||||
color: $accentCol;
|
color: var(--clr-primary);
|
||||||
|
|
||||||
&::after {
|
&::after {
|
||||||
content: '\027F6';
|
content: '\027F6';
|
||||||
@@ -505,7 +598,7 @@ export default defineComponent({
|
|||||||
font-size: 0.85em;
|
font-size: 0.85em;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen {
|
||||||
.timetable-item {
|
.timetable-item {
|
||||||
grid-template-columns: 1fr;
|
grid-template-columns: 1fr;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -40,36 +40,28 @@
|
|||||||
<span>
|
<span>
|
||||||
{{ $t('scenery.timetable-issued-date') }}
|
{{ $t('scenery.timetable-issued-date') }}
|
||||||
<b>
|
<b>
|
||||||
{{
|
{{ parseCreatedDate(timetableHistory, $i18n.locale) }}
|
||||||
localeDateTime(
|
|
||||||
timetableHistory.createdAt > timetableHistory.beginDate
|
|
||||||
? timetableHistory.beginDate
|
|
||||||
: timetableHistory.createdAt,
|
|
||||||
$i18n.locale
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
</b></span
|
|
||||||
>
|
|
||||||
<span v-if="timetableHistory.authorName">
|
|
||||||
{{ $t('scenery.timetable-issued-by') }}
|
|
||||||
<b>
|
|
||||||
<router-link
|
|
||||||
:to="`/journal/timetables?search-dispatcher=${timetableHistory.authorName}`"
|
|
||||||
>
|
|
||||||
{{ timetableHistory.authorName }}
|
|
||||||
</router-link>
|
|
||||||
</b>
|
</b>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span>
|
<span>
|
||||||
{{ $t('scenery.timetable-issued-for') }}
|
{{ $t('scenery.timetable-issued-for') }}
|
||||||
<b>
|
|
||||||
<router-link
|
<router-link
|
||||||
|
class="journal-link"
|
||||||
:to="`/journal/timetables?search-driver=${timetableHistory.driverName}`"
|
:to="`/journal/timetables?search-driver=${timetableHistory.driverName}`"
|
||||||
>
|
>
|
||||||
{{ timetableHistory.driverName }}
|
{{ timetableHistory.driverName }}
|
||||||
</router-link>
|
</router-link>
|
||||||
</b>
|
</span>
|
||||||
|
|
||||||
|
<span v-if="timetableHistory.authorName">
|
||||||
|
{{ $t('scenery.timetable-issued-by') }}
|
||||||
|
<router-link
|
||||||
|
class="journal-link"
|
||||||
|
:to="`/journal/timetables?search-dispatcher=${timetableHistory.authorName}`"
|
||||||
|
>
|
||||||
|
{{ timetableHistory.authorName }}
|
||||||
|
</router-link>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</span>
|
</span>
|
||||||
@@ -106,7 +98,7 @@ import { useApiStore } from '../../store/apiStore';
|
|||||||
import routerMixin from '../../mixins/routerMixin';
|
import routerMixin from '../../mixins/routerMixin';
|
||||||
import { useMainStore } from '../../store/mainStore';
|
import { useMainStore } from '../../store/mainStore';
|
||||||
|
|
||||||
const historyModeList = ['via', 'issuedFrom', 'terminatingAt'] as const;
|
const historyModeList = ['includesScenery', 'issuedFrom', 'via', 'terminatingAt'] as const;
|
||||||
type HistoryMode = (typeof historyModeList)[number];
|
type HistoryMode = (typeof historyModeList)[number];
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
@@ -131,17 +123,19 @@ export default defineComponent({
|
|||||||
dataStatus: Status.Data.Loading,
|
dataStatus: Status.Data.Loading,
|
||||||
DataStatus: Status.Data,
|
DataStatus: Status.Data,
|
||||||
|
|
||||||
checkedHistoryMode: 'via' as HistoryMode
|
checkedHistoryMode: 'includesScenery' as HistoryMode
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
async activated() {
|
async activated() {
|
||||||
|
this.checkedHistoryMode = 'includesScenery';
|
||||||
this.fetchAPIData();
|
this.fetchAPIData();
|
||||||
},
|
},
|
||||||
|
|
||||||
methods: {
|
methods: {
|
||||||
async fetchAPIData() {
|
async fetchAPIData() {
|
||||||
const stationName = this.$route.query['station'];
|
const stationName = this.$route.query['station'];
|
||||||
|
this.dataStatus = Status.Data.Loading;
|
||||||
|
|
||||||
if (!stationName) {
|
if (!stationName) {
|
||||||
this.historyList = [];
|
this.historyList = [];
|
||||||
@@ -152,6 +146,7 @@ export default defineComponent({
|
|||||||
const requestFilters: Record<string, any> = {};
|
const requestFilters: Record<string, any> = {};
|
||||||
requestFilters[this.checkedHistoryMode] = stationName.toString();
|
requestFilters[this.checkedHistoryMode] = stationName.toString();
|
||||||
requestFilters.countLimit = 30;
|
requestFilters.countLimit = 30;
|
||||||
|
requestFilters['returnType'] = 'short';
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const response: API.TimetableHistory.Response = await (
|
const response: API.TimetableHistory.Response = await (
|
||||||
@@ -165,12 +160,12 @@ export default defineComponent({
|
|||||||
this.dataStatus = Status.Data.Loaded;
|
this.dataStatus = Status.Data.Loaded;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(error);
|
console.error(error);
|
||||||
|
this.dataStatus = Status.Data.Error;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
checkHistoryMode(mode: HistoryMode) {
|
checkHistoryMode(mode: HistoryMode) {
|
||||||
this.checkedHistoryMode = mode;
|
this.checkedHistoryMode = mode;
|
||||||
this.dataStatus = Status.Data.Loading;
|
|
||||||
this.fetchAPIData();
|
this.fetchAPIData();
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -181,6 +176,18 @@ export default defineComponent({
|
|||||||
[`search-${this.checkedHistoryMode}`]: this.station?.name || this.onlineScenery?.name
|
[`search-${this.checkedHistoryMode}`]: this.station?.name || this.onlineScenery?.name
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
parseCreatedDate(timetable: API.TimetableHistory.Data, locale: string) {
|
||||||
|
const createdDate =
|
||||||
|
timetable.createdAt > timetable.beginDate
|
||||||
|
? new Date(timetable.beginDate)
|
||||||
|
: new Date(timetable.createdAt);
|
||||||
|
|
||||||
|
return createdDate.toLocaleString(locale == 'pl' ? 'pl-PL' : 'en-GB', {
|
||||||
|
timeStyle: 'short',
|
||||||
|
dateStyle: 'medium'
|
||||||
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
components: { Loading }
|
components: { Loading }
|
||||||
@@ -188,8 +195,8 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/sceneryViewTables.scss';
|
@use '../../styles/scenery-history-table';
|
||||||
|
|
||||||
.scenery-timetables-history {
|
.scenery-timetables-history {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@@ -215,7 +222,15 @@ export default defineComponent({
|
|||||||
|
|
||||||
button {
|
button {
|
||||||
padding: 0.35em;
|
padding: 0.35em;
|
||||||
min-width: 120px;
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.journal-link {
|
||||||
|
font-weight: bold;
|
||||||
|
color: #eee;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
color: var(--clr-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,11 +1,18 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="general-status">
|
<div class="general-status">
|
||||||
<span
|
<router-link
|
||||||
|
v-if="computedScheduledTrain.stationNameHref"
|
||||||
|
:to="`/scenery?station=${computedScheduledTrain.stationNameHref}`"
|
||||||
:class="computedScheduledTrain.status"
|
:class="computedScheduledTrain.status"
|
||||||
:title="computedScheduledTrain.stopStatusDescription"
|
v-html="computedScheduledTrain.stopStatusIndicator"
|
||||||
>
|
>
|
||||||
{{ computedScheduledTrain.stopStatusIndicator }}
|
</router-link>
|
||||||
</span>
|
|
||||||
|
<span
|
||||||
|
v-else
|
||||||
|
:class="computedScheduledTrain.status"
|
||||||
|
v-html="computedScheduledTrain.stopStatusIndicator"
|
||||||
|
></span>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
@@ -24,57 +31,67 @@ export default defineComponent({
|
|||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
computedScheduledTrain() {
|
computedScheduledTrain() {
|
||||||
const { prevDepartureLine, prevStationName, nextArrivalLine, nextStationName, status } =
|
const { status, prevElement, currentElement, nextElement } = this.sceneryTimetableRow;
|
||||||
this.sceneryTimetableRow;
|
|
||||||
|
|
||||||
const prevDepartureIndicator = prevDepartureLine
|
let stopStatusIndicator = '';
|
||||||
? `(${prevDepartureLine}) ${prevStationName}`
|
let stationNameHref = '';
|
||||||
: '---';
|
|
||||||
const nextArrivalIndicator = nextArrivalLine
|
|
||||||
? `(${nextArrivalLine}) ${nextStationName}`
|
|
||||||
: '---';
|
|
||||||
|
|
||||||
let stopStatusDescription = '',
|
|
||||||
stopStatusIndicator = '';
|
|
||||||
|
|
||||||
switch (status) {
|
switch (status) {
|
||||||
case StopStatus.ARRIVING:
|
case StopStatus.ARRIVING:
|
||||||
stopStatusIndicator = `${this.$t('timetables.from')}: ${prevDepartureIndicator}`;
|
if (prevElement) {
|
||||||
stopStatusDescription = this.$t('timetables.desc-arriving', {
|
stopStatusIndicator = this.$t('timetables.desc-arriving', {
|
||||||
prevStationName,
|
prevStationName: prevElement?.stationName ?? '',
|
||||||
prevDepartureLine
|
prevDepartureLine: prevElement?.departureRouteExt ?? ''
|
||||||
});
|
});
|
||||||
|
|
||||||
|
stationNameHref = prevElement?.stationName ?? '';
|
||||||
|
} else {
|
||||||
|
stopStatusIndicator = this.$t('timetables.desc-beginning');
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case StopStatus.ONLINE:
|
case StopStatus.ONLINE:
|
||||||
case StopStatus.STOPPED:
|
case StopStatus.STOPPED:
|
||||||
stopStatusIndicator = nextArrivalLine
|
stopStatusIndicator = nextElement?.arrivalRouteExt
|
||||||
? `${this.$t('timetables.to')}: ${nextArrivalIndicator}`
|
? this.$t(`timetables.desc-${status}`, {
|
||||||
: `${this.$t('timetables.desc-end')}`;
|
nextStationName: nextElement?.stationName,
|
||||||
stopStatusDescription = nextArrivalLine
|
nextArrivalLine: nextElement?.arrivalRouteExt
|
||||||
? this.$t(`timetables.desc-${status}`, { nextStationName, nextArrivalLine })
|
})
|
||||||
: '';
|
: this.$t(`timetables.desc-end`);
|
||||||
|
|
||||||
|
stationNameHref = nextElement?.stationName ?? '';
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case StopStatus.DEPARTED:
|
case StopStatus.DEPARTED:
|
||||||
stopStatusIndicator = `${this.$t('timetables.to')}: ${nextArrivalIndicator}`;
|
if (!nextElement?.stationName) {
|
||||||
stopStatusDescription = this.$t('timetables.desc-departed', {
|
stopStatusIndicator = this.$t('timetables.desc-departed-ends', {
|
||||||
nextStationName,
|
nextStationName: currentElement.stationName
|
||||||
nextArrivalLine
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
stationNameHref = nextElement?.stationName ?? '';
|
||||||
|
} else {
|
||||||
|
stopStatusIndicator = this.$t('timetables.desc-departed', {
|
||||||
|
nextStationName: nextElement?.stationName ?? currentElement.stationName,
|
||||||
|
nextArrivalLine: nextElement?.arrivalRouteExt
|
||||||
|
});
|
||||||
|
|
||||||
|
stationNameHref = nextElement?.stationName ?? '';
|
||||||
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case StopStatus.DEPARTED_AWAY:
|
case StopStatus.DEPARTED_AWAY:
|
||||||
stopStatusIndicator = `${this.$t('timetables.to')}: ${nextArrivalIndicator}`;
|
stopStatusIndicator = this.$t('timetables.desc-departed-away', {
|
||||||
stopStatusDescription = this.$t('timetables.desc-departed-away', {
|
nextStationName: nextElement?.stationName,
|
||||||
nextStationName,
|
nextArrivalLine: nextElement?.arrivalRouteExt
|
||||||
nextArrivalLine
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
stationNameHref = nextElement?.stationName ?? '';
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case StopStatus.TERMINATED:
|
case StopStatus.TERMINATED:
|
||||||
stopStatusIndicator = `X ${this.$t('timetables.desc-terminated')}`;
|
stopStatusIndicator = this.$t('timetables.desc-terminated');
|
||||||
stopStatusDescription = this.$t('timetables.desc-terminated');
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@@ -82,10 +99,18 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
...this.sceneryTimetableRow,
|
...this.sceneryTimetableRow,
|
||||||
stopStatusDescription,
|
stationNameHref,
|
||||||
stopStatusIndicator
|
stopStatusIndicator
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
navigateToScenery(sceneryName?: string) {
|
||||||
|
if (!sceneryName) return;
|
||||||
|
|
||||||
|
this.$router.push(`/scenery?station=${sceneryName}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
@@ -94,32 +119,28 @@ export default defineComponent({
|
|||||||
.general-status {
|
.general-status {
|
||||||
margin-top: 0.5em;
|
margin-top: 0.5em;
|
||||||
|
|
||||||
span.arriving {
|
& > .arriving {
|
||||||
color: #ccc;
|
color: #ccc;
|
||||||
}
|
}
|
||||||
|
|
||||||
span.departed {
|
& > .departed {
|
||||||
color: lime;
|
color: lime;
|
||||||
font-weight: bold;
|
|
||||||
|
|
||||||
&-away {
|
&-away {
|
||||||
font-weight: bold;
|
|
||||||
color: #5ecc5e;
|
color: #5ecc5e;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
span.stopped {
|
& > .stopped {
|
||||||
color: #ffa600;
|
color: #ffa600;
|
||||||
font-weight: bold;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
span.online {
|
& > .online {
|
||||||
color: gold;
|
color: gold;
|
||||||
}
|
}
|
||||||
|
|
||||||
span.terminated {
|
& > .terminated {
|
||||||
color: salmon;
|
color: salmon;
|
||||||
font-weight: bold;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,13 +1,10 @@
|
|||||||
import { StopStatus, Train, TrainStop } from '../../typings/common';
|
import { StopStatus, TimetablePathElement, Train, TrainStop } from '../../typings/common';
|
||||||
|
|
||||||
export interface SceneryTimetableRow {
|
export interface SceneryTimetableRow {
|
||||||
checkpointStop: TrainStop;
|
checkpointStop: TrainStop;
|
||||||
train: Train;
|
train: Train;
|
||||||
prevDepartureLine: string | null;
|
prevElement: TimetablePathElement | null;
|
||||||
nextArrivalLine: string | null;
|
nextElement: TimetablePathElement | null;
|
||||||
departureLine: string | null;
|
currentElement: TimetablePathElement;
|
||||||
arrivingLine: string | null;
|
|
||||||
prevStationName: string | null;
|
|
||||||
nextStationName: string | null;
|
|
||||||
status: StopStatus;
|
status: StopStatus;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -66,8 +66,6 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/variables.scss';
|
|
||||||
|
|
||||||
label {
|
label {
|
||||||
position: relative;
|
position: relative;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
@@ -98,7 +96,7 @@ label {
|
|||||||
}
|
}
|
||||||
|
|
||||||
&:focus-visible + span {
|
&:focus-visible + span {
|
||||||
outline: 1px solid $accentCol;
|
outline: 1px solid var(--clr-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,9 +21,7 @@
|
|||||||
<template v-else>{{ $t('filters.no-changed-filters') }}</template>
|
<template v-else>{{ $t('filters.no-changed-filters') }}</template>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<section class="card_sceneries-search">
|
<section class="card_input-search">
|
||||||
<h3 class="section-header">{{ $t('filters.sceneries-search') }}</h3>
|
|
||||||
|
|
||||||
<datalist id="sceneries">
|
<datalist id="sceneries">
|
||||||
<option
|
<option
|
||||||
v-for="scenery in sortedStationList"
|
v-for="scenery in sortedStationList"
|
||||||
@@ -32,7 +30,6 @@
|
|||||||
></option>
|
></option>
|
||||||
</datalist>
|
</datalist>
|
||||||
|
|
||||||
<form action="javascript:void(0);" @submit="handleSceneriesInput">
|
|
||||||
<input
|
<input
|
||||||
v-model="chosenSearchScenery"
|
v-model="chosenSearchScenery"
|
||||||
id="scenery-search"
|
id="scenery-search"
|
||||||
@@ -42,8 +39,51 @@
|
|||||||
@blur="preventKeyDown = false"
|
@blur="preventKeyDown = false"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<button class="btn--action">{{ $t('filters.search-button-title') }}</button>
|
<button class="btn--action" @click="handleSceneriesInput">
|
||||||
</form>
|
{{ $t('filters.search-button-title') }}
|
||||||
|
</button>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="card_input-search authors">
|
||||||
|
<datalist id="authors" name="authors">
|
||||||
|
<option v-for="(author, i) in authorsOptions" :key="i" :value="author"></option>
|
||||||
|
</datalist>
|
||||||
|
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
id="author"
|
||||||
|
list="authors"
|
||||||
|
name="authors"
|
||||||
|
v-model="filters['authors']"
|
||||||
|
:placeholder="$t('filters.authors-placeholder')"
|
||||||
|
@focus="preventKeyDown = true"
|
||||||
|
@blur="preventKeyDown = false"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<button class="btn--action btn--image" @click="resetAuthorsInput">
|
||||||
|
<img src="/images/icon-exit.svg" alt="reset authors search" />
|
||||||
|
</button>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="card_input-search">
|
||||||
|
<datalist id="projects" name="projects">
|
||||||
|
<option v-for="(project, i) in projectsOptions" :key="i" :value="project"></option>
|
||||||
|
</datalist>
|
||||||
|
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
id="projects"
|
||||||
|
list="projects"
|
||||||
|
name="projects"
|
||||||
|
v-model="filters['projects']"
|
||||||
|
:placeholder="$t('filters.projects-placeholder')"
|
||||||
|
@focus="preventKeyDown = true"
|
||||||
|
@blur="preventKeyDown = false"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<button class="btn--action btn--image" @click="resetProjectsInput">
|
||||||
|
<img src="/images/icon-exit.svg" alt="reset projects search" />
|
||||||
|
</button>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section class="card_options">
|
<section class="card_options">
|
||||||
@@ -97,31 +137,8 @@
|
|||||||
</span>
|
</span>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section class="card_authors-search">
|
|
||||||
<h3 class="section-header">{{ $t('filters.authors-search') }}</h3>
|
|
||||||
|
|
||||||
<datalist id="authors" name="authors">
|
|
||||||
<option v-for="(author, i) in authorsHint" :key="i" :value="author"></option>
|
|
||||||
</datalist>
|
|
||||||
|
|
||||||
<form action="javascript:void(0);" @submit="handleAuthorsInput">
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
id="author"
|
|
||||||
list="authors"
|
|
||||||
name="authors"
|
|
||||||
v-model="authors"
|
|
||||||
:placeholder="$t('filters.authors-placeholder')"
|
|
||||||
@focus="preventKeyDown = true"
|
|
||||||
@blur="preventKeyDown = false"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<button class="btn--action">{{ $t('filters.search-button-title') }}</button>
|
|
||||||
</form>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<section class="card_sliders">
|
<section class="card_sliders">
|
||||||
<div class="slider" v-for="(slider, i) in initSliders" :key="i">
|
<div class="slider" v-for="(slider, i) in sliderStates" :key="i">
|
||||||
<input
|
<input
|
||||||
class="slider-input"
|
class="slider-input"
|
||||||
type="range"
|
type="range"
|
||||||
@@ -130,7 +147,7 @@
|
|||||||
:min="slider.minRange"
|
:min="slider.minRange"
|
||||||
:max="slider.maxRange"
|
:max="slider.maxRange"
|
||||||
:step="slider.step"
|
:step="slider.step"
|
||||||
v-model="filters[slider.id]"
|
v-model.number="filters[slider.id]"
|
||||||
/>
|
/>
|
||||||
<span class="slider-value">{{ filters[slider.id] }}</span>
|
<span class="slider-value">{{ filters[slider.id] }}</span>
|
||||||
<div class="slider-content">
|
<div class="slider-content">
|
||||||
@@ -178,7 +195,7 @@ import StorageManager from '../../managers/storageManager';
|
|||||||
|
|
||||||
import {
|
import {
|
||||||
filtersSections,
|
filtersSections,
|
||||||
initSliders,
|
sliderStates,
|
||||||
initFilters,
|
initFilters,
|
||||||
getChangedFilters
|
getChangedFilters
|
||||||
} from '../../managers/stationFilterManager';
|
} from '../../managers/stationFilterManager';
|
||||||
@@ -197,10 +214,11 @@ export default defineComponent({
|
|||||||
saveOptions: false,
|
saveOptions: false,
|
||||||
|
|
||||||
filtersSections,
|
filtersSections,
|
||||||
initSliders,
|
sliderStates,
|
||||||
|
|
||||||
minimumHours: 0,
|
minimumHours: 0,
|
||||||
authors: '',
|
authorSearchFilter: '',
|
||||||
|
projectSearchFilter: '',
|
||||||
|
|
||||||
currentRegion: { id: '', value: '' },
|
currentRegion: { id: '', value: '' },
|
||||||
|
|
||||||
@@ -255,11 +273,7 @@ export default defineComponent({
|
|||||||
.sort((s1, s2) => (s1.name > s2.name ? 1 : -1));
|
.sort((s1, s2) => (s1.name > s2.name ? 1 : -1));
|
||||||
},
|
},
|
||||||
|
|
||||||
currentOptionsActive() {
|
authorsOptions() {
|
||||||
return true;
|
|
||||||
},
|
|
||||||
|
|
||||||
authorsHint() {
|
|
||||||
return this.store.stationList
|
return this.store.stationList
|
||||||
.reduce((acc, station) => {
|
.reduce((acc, station) => {
|
||||||
station.generalInfo?.authors?.forEach((author) => {
|
station.generalInfo?.authors?.forEach((author) => {
|
||||||
@@ -270,6 +284,17 @@ export default defineComponent({
|
|||||||
return acc;
|
return acc;
|
||||||
}, [] as string[])
|
}, [] as string[])
|
||||||
.sort((a, b) => a.localeCompare(b));
|
.sort((a, b) => a.localeCompare(b));
|
||||||
|
},
|
||||||
|
|
||||||
|
projectsOptions() {
|
||||||
|
return this.store.stationList
|
||||||
|
.reduce((acc, station) => {
|
||||||
|
if (!station.generalInfo || !station.generalInfo.project || station.generalInfo.hidden) return acc;
|
||||||
|
if (!acc.includes(station.generalInfo.project.trim())) acc.push(station.generalInfo.project.trim());
|
||||||
|
|
||||||
|
return acc;
|
||||||
|
}, [] as string[])
|
||||||
|
.sort((a, b) => a.localeCompare(b));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -294,8 +319,12 @@ export default defineComponent({
|
|||||||
this.scrollTop = (e.target as HTMLElement).scrollTop;
|
this.scrollTop = (e.target as HTMLElement).scrollTop;
|
||||||
},
|
},
|
||||||
|
|
||||||
handleAuthorsInput() {
|
resetAuthorsInput() {
|
||||||
this.filters['authors'] = this.authors;
|
this.filters['authors'] = this.authorSearchFilter;
|
||||||
|
},
|
||||||
|
|
||||||
|
resetProjectsInput() {
|
||||||
|
this.filters['projects'] = this.projectSearchFilter;
|
||||||
},
|
},
|
||||||
|
|
||||||
handleSceneriesInput() {
|
handleSceneriesInput() {
|
||||||
@@ -340,7 +369,7 @@ export default defineComponent({
|
|||||||
|
|
||||||
// Reset local model values
|
// Reset local model values
|
||||||
this.minimumHours = 0;
|
this.minimumHours = 0;
|
||||||
this.authors = '';
|
this.authorSearchFilter = '';
|
||||||
|
|
||||||
// Reset global filters
|
// Reset global filters
|
||||||
Object.keys(this.filters).forEach((filterKey) => {
|
Object.keys(this.filters).forEach((filterKey) => {
|
||||||
@@ -379,10 +408,9 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/card';
|
@use '../../styles/card';
|
||||||
@import '../../styles/animations';
|
@use '../../styles/animations';
|
||||||
@import '../../styles/variables';
|
|
||||||
|
|
||||||
h3.section-header {
|
h3.section-header {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
@@ -412,11 +440,6 @@ h3.section-header {
|
|||||||
.card_controls {
|
.card_controls {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
|
|
||||||
input {
|
|
||||||
border-radius: 0.5em 0.5em 0 0;
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.card_content {
|
.card_content {
|
||||||
@@ -432,7 +455,7 @@ h3.section-header {
|
|||||||
.card_title {
|
.card_title {
|
||||||
font-size: 2em;
|
font-size: 2em;
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
color: $accentCol;
|
color: var(--clr-primary);
|
||||||
|
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
@@ -453,7 +476,7 @@ h3.section-header {
|
|||||||
span {
|
span {
|
||||||
min-width: 120px;
|
min-width: 120px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: $accentCol;
|
color: var(--clr-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
button {
|
button {
|
||||||
@@ -462,27 +485,23 @@ h3.section-header {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.card_authors-search,
|
.card_input-search {
|
||||||
.card_sceneries-search {
|
|
||||||
margin: 1em 0;
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
|
||||||
form {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
width: 100%;
|
|
||||||
margin-top: 1em;
|
button {
|
||||||
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
input {
|
input {
|
||||||
width: 70%;
|
width: 100%;
|
||||||
max-width: 400px;
|
|
||||||
padding: 0.5em;
|
padding: 0.5em;
|
||||||
outline: 1px solid white;
|
border: 1px solid #aaa;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.authors {
|
||||||
|
margin-top: 1em;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -527,7 +546,7 @@ h3.section-header {
|
|||||||
}
|
}
|
||||||
|
|
||||||
&:focus-visible + span {
|
&:focus-visible + span {
|
||||||
outline: 1px solid $accentCol;
|
outline: 1px solid var(--clr-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -567,15 +586,17 @@ h3.section-header {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.slider {
|
.slider {
|
||||||
display: flex;
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 50px 1fr;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 0.25em;
|
gap: 0.25em;
|
||||||
|
|
||||||
margin-bottom: 1em;
|
margin-bottom: 1em;
|
||||||
|
|
||||||
&-value {
|
&-value {
|
||||||
color: $accentCol;
|
color: var(--clr-primary);
|
||||||
padding: 0.1em 0.2em;
|
padding: 0.1em 0.2em;
|
||||||
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
&-input {
|
&-input {
|
||||||
@@ -602,13 +623,14 @@ h3.section-header {
|
|||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
|
|
||||||
background: white;
|
background: white;
|
||||||
border: 4px solid $accentCol;
|
border: 3px solid var(--clr-primary);
|
||||||
|
background-color: #333;
|
||||||
|
|
||||||
@include smallScreen() {
|
@include responsive.smallScreen {
|
||||||
width: 15px;
|
width: 15px;
|
||||||
height: 15px;
|
height: 15px;
|
||||||
margin-top: -5px;
|
margin-top: -5px;
|
||||||
border: 3px solid $accentCol;
|
border: 3px solid var(--clr-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -619,14 +641,14 @@ h3.section-header {
|
|||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
|
|
||||||
background: white;
|
background: white;
|
||||||
border: 4px solid $accentCol;
|
border: 4px solid var(--clr-primary);
|
||||||
|
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
|
||||||
@include smallScreen() {
|
@include responsive.smallScreen {
|
||||||
width: 1em;
|
width: 1em;
|
||||||
height: 1em;
|
height: 1em;
|
||||||
border: 3px solid $accentCol;
|
border: 3px solid var(--clr-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -656,14 +678,23 @@ h3.section-header {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen {
|
||||||
.slider {
|
.slider {
|
||||||
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
|
||||||
&-input {
|
&-input {
|
||||||
width: 90%;
|
width: 90%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&-content {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.card_controls > button > p {
|
||||||
|
display: none;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -8,22 +8,22 @@
|
|||||||
|
|
||||||
<button class="filter-button btn--filled btn--image" @click="toggleDropdown" ref="button">
|
<button class="filter-button btn--filled btn--image" @click="toggleDropdown" ref="button">
|
||||||
<img src="/images/icon-stats.svg" alt="Open filters icon" />
|
<img src="/images/icon-stats.svg" alt="Open filters icon" />
|
||||||
<!-- {{ $t('train-stats.stats-button') }} -->
|
<span>{{ $t('station-stats.stats-button') }}</span>
|
||||||
<span>STATYSTYKI</span>
|
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<transition name="dropdown-anim">
|
<transition name="dropdown-anim">
|
||||||
<div class="dropdown_wrapper" v-if="showDropdown">
|
<div class="dropdown_wrapper" v-if="showDropdown">
|
||||||
<div>
|
<div>
|
||||||
<h1 class="text--primary">
|
<h2 class="stats-title text--primary">
|
||||||
<img src="/images/icon-stats.svg" alt="Open filters icon" />
|
<img src="/images/icon-stats.svg" alt="Open filters icon" height="28" />
|
||||||
{{ $t('train-stats.title') }}
|
{{ $t('station-stats.title') }}
|
||||||
</h1>
|
</h2>
|
||||||
|
|
||||||
<hr style="margin: 0.5em 0" />
|
<hr style="margin: 0.5em 0" />
|
||||||
|
|
||||||
<ul class="stats-list">
|
<div v-if="uFactor > -1 || avgTimetableCount > -1 || trackCount.all > 0">
|
||||||
<li>
|
<div class="badges-container">
|
||||||
|
<div class="badge stat-badge">
|
||||||
<span>
|
<span>
|
||||||
{{ $t('station-stats.u-factor') }}
|
{{ $t('station-stats.u-factor') }}
|
||||||
<a
|
<a
|
||||||
@@ -33,33 +33,81 @@
|
|||||||
>(?)</a
|
>(?)</a
|
||||||
>:
|
>:
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
|
<span>
|
||||||
<b class="u-factor" :style="calculateFactorStyle()">
|
<b class="u-factor" :style="calculateFactorStyle()">
|
||||||
{{ uFactor.toFixed(2) }}
|
{{ uFactor >= 0 ? uFactor.toFixed(2) : '---' }}
|
||||||
</b>
|
</b>
|
||||||
</li>
|
</span>
|
||||||
<li>
|
</div>
|
||||||
{{ $t('station-stats.avg-timetable-count') }}
|
|
||||||
<b>{{ avgTimetableCount.toFixed(2) }}</b>
|
<div class="badge stat-badge">
|
||||||
</li>
|
<span>{{ $t('station-stats.avg-timetable-count') }}</span>
|
||||||
<li>
|
<span>
|
||||||
{{ $t('station-stats.single-track-count') }}
|
<b>{{ avgTimetableCount >= 0 ? avgTimetableCount.toFixed(2) : '---' }}</b>
|
||||||
<b>{{ trackCount.oneWay }}</b> (<b>{{ trackCount.oneWayElectric }} ⚡</b>)
|
</span>
|
||||||
</li>
|
</div>
|
||||||
<li>
|
</div>
|
||||||
{{ $t('station-stats.double-track-count') }}
|
|
||||||
<b>{{ trackCount.twoWay }}</b>
|
<hr style="margin: 0.5em 0" />
|
||||||
(<b>{{ trackCount.twoWayElectric }} ⚡</b>)
|
|
||||||
</li>
|
<div class="badges-container">
|
||||||
<li>
|
<div class="badge stat-badge">
|
||||||
{{ $t('station-stats.cross-sceneries') }}
|
<span>{{ $t('station-stats.single-track-count') }}</span>
|
||||||
|
<span>
|
||||||
|
<b> {{ trackCount.oneWay }}</b> (<b>{{ trackCount.oneWayElectric }} ⚡</b>)
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="badge stat-badge">
|
||||||
|
<span>{{ $t('station-stats.double-track-count') }}</span>
|
||||||
|
<span>
|
||||||
|
<b>{{ trackCount.twoWay }}</b> (<b>{{ trackCount.twoWayElectric }} ⚡</b>)
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="badge stat-badge">
|
||||||
|
<span> {{ $t('station-stats.cross-sceneries') }}</span>
|
||||||
|
<span>
|
||||||
<b>{{ trackCount.crossTrack }}</b> (<b>{{ trackCount.crossTrackElectric }} ⚡</b>)
|
<b>{{ trackCount.crossTrack }}</b> (<b>{{ trackCount.crossTrackElectric }} ⚡</b>)
|
||||||
</li>
|
</span>
|
||||||
<li>
|
</div>
|
||||||
{{ $t('station-stats.open-spawns') }} <b>{{ spawnCount.passenger }}</b> - PAS /
|
</div>
|
||||||
<b>{{ spawnCount.freight }}</b> - TOW / <b>{{ spawnCount.loco }}</b> - LUZ /
|
|
||||||
<b>{{ spawnCount.all }}</b> - ALL
|
<hr style="margin: 0.5em 0" />
|
||||||
</li>
|
|
||||||
</ul>
|
<div class="badges-container">
|
||||||
|
<div class="badge stat-badge">
|
||||||
|
<span> {{ $t('station-stats.open-spawns-all') }}</span>
|
||||||
|
<span>
|
||||||
|
<b>{{ spawnCount.all }}</b>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="badge stat-badge">
|
||||||
|
<span> {{ $t('station-stats.open-spawns-pas') }}</span>
|
||||||
|
<span>
|
||||||
|
<b>{{ spawnCount.passenger }}</b>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="badge stat-badge">
|
||||||
|
<span> {{ $t('station-stats.open-spawns-freight') }}</span>
|
||||||
|
<span>
|
||||||
|
<b>{{ spawnCount.freight }}</b>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="badge stat-badge">
|
||||||
|
<span> {{ $t('station-stats.open-spawns-loco') }}</span>
|
||||||
|
<span>
|
||||||
|
<b>{{ spawnCount.loco }}</b>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="no-data" v-else>{{ $t('station-stats.no-stats') }}</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div tabindex="0" @focus="() => (showDropdown = false)"></div>
|
<div tabindex="0" @focus="() => (showDropdown = false)"></div>
|
||||||
@@ -86,9 +134,9 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
calculateFactorStyle() {
|
calculateFactorStyle() {
|
||||||
if (this.uFactor == 0) return '';
|
if (this.uFactor <= 0) return '';
|
||||||
|
|
||||||
const norm = this.uFactor == 0 ? 1 : Math.max(Math.min(this.uFactor / 2, 1), 0);
|
const norm = Math.max(Math.min(this.uFactor / 2, 1), 0);
|
||||||
const lerp = 120 * norm;
|
const lerp = 120 * norm;
|
||||||
|
|
||||||
return `color: hsl(${lerp}, 100%, 60%)`;
|
return `color: hsl(${lerp}, 100%, 60%)`;
|
||||||
@@ -105,7 +153,7 @@ export default defineComponent({
|
|||||||
(train) => train.region == this.mainStore.region.id
|
(train) => train.region == this.mainStore.region.id
|
||||||
);
|
);
|
||||||
|
|
||||||
return activeDispatchers.length != 0 ? activeTrains.length / activeDispatchers.length : 0;
|
return activeDispatchers.length != 0 ? activeTrains.length / activeDispatchers.length : -1;
|
||||||
},
|
},
|
||||||
|
|
||||||
avgTimetableCount() {
|
avgTimetableCount() {
|
||||||
@@ -118,7 +166,7 @@ export default defineComponent({
|
|||||||
return acc;
|
return acc;
|
||||||
}, 0);
|
}, 0);
|
||||||
|
|
||||||
if (regionSceneries.length == 0) return 0;
|
if (regionSceneries.length == 0) return -1;
|
||||||
|
|
||||||
return timetableCountSum / regionSceneries.length;
|
return timetableCountSum / regionSceneries.length;
|
||||||
},
|
},
|
||||||
@@ -135,6 +183,8 @@ export default defineComponent({
|
|||||||
(acc, st) => {
|
(acc, st) => {
|
||||||
const { routes } = st.generalInfo!;
|
const { routes } = st.generalInfo!;
|
||||||
|
|
||||||
|
acc.all++;
|
||||||
|
|
||||||
if (
|
if (
|
||||||
routes.single.filter((r) => !r.isInternal).length > 0 &&
|
routes.single.filter((r) => !r.isInternal).length > 0 &&
|
||||||
routes.double.filter((r) => !r.isInternal).length > 0
|
routes.double.filter((r) => !r.isInternal).length > 0
|
||||||
@@ -163,7 +213,8 @@ export default defineComponent({
|
|||||||
twoWay: 0,
|
twoWay: 0,
|
||||||
twoWayElectric: 0,
|
twoWayElectric: 0,
|
||||||
crossTrack: 0,
|
crossTrack: 0,
|
||||||
crossTrackElectric: 0
|
crossTrackElectric: 0,
|
||||||
|
all: 0
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@@ -190,15 +241,18 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/dropdown.scss';
|
@use '../../styles/dropdown';
|
||||||
@import '../../styles/badge.scss';
|
@use '../../styles/badge';
|
||||||
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
h1 img {
|
.stats-title img {
|
||||||
vertical-align: text-bottom;
|
vertical-align: text-bottom;
|
||||||
}
|
}
|
||||||
|
|
||||||
h3 {
|
.badges-container {
|
||||||
margin: 0.5em 0;
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.u-factor {
|
.u-factor {
|
||||||
@@ -219,18 +273,17 @@ h3 {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ul.stats-list {
|
.no-data {
|
||||||
list-style: disc;
|
font-size: 1.1em;
|
||||||
padding-left: 1em;
|
color: #ccc;
|
||||||
margin-top: 1em;
|
|
||||||
|
|
||||||
& > li {
|
|
||||||
margin: 0.25em 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen {
|
||||||
.filter-button span {
|
.stats-title {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.filter-button > span {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -33,12 +33,12 @@
|
|||||||
class="header-image"
|
class="header-image"
|
||||||
:class="headerName"
|
:class="headerName"
|
||||||
>
|
>
|
||||||
<span class="header_wrapper">
|
<span
|
||||||
<img
|
class="header_wrapper"
|
||||||
:src="`/images/icon-${headerName}.svg`"
|
data-tooltip-type="BaseTooltip"
|
||||||
:alt="headerName"
|
:data-tooltip-content="$t(`sceneries.headers.${headerName}`)"
|
||||||
:title="$t(`sceneries.headers.${headerName}`)"
|
>
|
||||||
/>
|
<img :src="`/images/icon-${headerName}.svg`" :alt="headerName" />
|
||||||
|
|
||||||
<img
|
<img
|
||||||
class="sort-icon"
|
class="sort-icon"
|
||||||
@@ -76,37 +76,49 @@
|
|||||||
station.generalInfo.availability != 'nonPublic' &&
|
station.generalInfo.availability != 'nonPublic' &&
|
||||||
station.generalInfo.availability != 'unavailable'
|
station.generalInfo.availability != 'unavailable'
|
||||||
"
|
"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="`${$t(`sceneries.info.${station.generalInfo.availability}`)} (${$t(
|
||||||
|
'sceneries.info.req-level',
|
||||||
|
{ lvl: station.generalInfo.reqLevel },
|
||||||
|
station.generalInfo.reqLevel
|
||||||
|
)})`"
|
||||||
:style="calculateExpStyle(station.generalInfo.reqLevel)"
|
:style="calculateExpStyle(station.generalInfo.reqLevel)"
|
||||||
>
|
>
|
||||||
{{ station.generalInfo.reqLevel >= 2 ? station.generalInfo.reqLevel : 'L' }}
|
{{ station.generalInfo.reqLevel >= 2 ? station.generalInfo.reqLevel : 'L' }}
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span v-else-if="station.generalInfo.availability == 'abandoned'">
|
<span
|
||||||
<img
|
v-else-if="station.generalInfo.availability == 'abandoned'"
|
||||||
src="/images/icon-abandoned.svg"
|
data-tooltip-type="BaseTooltip"
|
||||||
alt="non-public"
|
:data-tooltip-content="$t('sceneries.info.abandoned')"
|
||||||
:title="$t('sceneries.info.abandoned')"
|
>
|
||||||
/>
|
<img src="/images/icon-abandoned.svg" alt="non-public" />
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span v-else-if="station.generalInfo.availability == 'nonPublic'">
|
<span
|
||||||
<img
|
v-else-if="station.generalInfo.availability == 'nonPublic'"
|
||||||
src="/images/icon-lock.svg"
|
data-tooltip-type="BaseTooltip"
|
||||||
alt="non-public"
|
:data-tooltip-content="$t('sceneries.info.non-public')"
|
||||||
:title="$t('sceneries.info.non-public')"
|
>
|
||||||
/>
|
<img src="/images/icon-lock.svg" alt="non-public" />
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span v-else>
|
<span
|
||||||
<img
|
v-else
|
||||||
src="/images/icon-unavailable.svg"
|
data-tooltip-type="BaseTooltip"
|
||||||
alt="unavailable"
|
:data-tooltip-content="$t('sceneries.info.unavailable')"
|
||||||
:title="$t('sceneries.info.unavailable')"
|
>
|
||||||
/>
|
<img src="/images/icon-unavailable.svg" alt="unavailable" />
|
||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span v-else> ? </span>
|
<span
|
||||||
|
v-else
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('sceneries.info.unknown')"
|
||||||
|
>
|
||||||
|
?
|
||||||
|
</span>
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td class="station-status">
|
<td class="station-status">
|
||||||
@@ -124,8 +136,8 @@
|
|||||||
data-tooltip-type="DonatorTooltip"
|
data-tooltip-type="DonatorTooltip"
|
||||||
:data-tooltip-content="$t('donations.dispatcher-message')"
|
:data-tooltip-content="$t('donations.dispatcher-message')"
|
||||||
>
|
>
|
||||||
<img src="/images/icon-diamond.svg" alt="" />
|
<img src="/images/icon-diamond.svg" alt="donator diamond icon" />
|
||||||
{{ station.onlineInfo.dispatcherName }}
|
<span class="text--donator"> {{ station.onlineInfo.dispatcherName }}</span>
|
||||||
</b>
|
</b>
|
||||||
|
|
||||||
<div v-else>
|
<div v-else>
|
||||||
@@ -153,7 +165,8 @@
|
|||||||
<span
|
<span
|
||||||
v-if="station.generalInfo.routes.singleElectrifiedNames.length != 0"
|
v-if="station.generalInfo.routes.singleElectrifiedNames.length != 0"
|
||||||
class="track catenary"
|
class="track catenary"
|
||||||
:title="`${$t('sceneries.info.single-track-routes-catenary')}${
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="`${$t('sceneries.info.single-track-routes-catenary')}${
|
||||||
station.generalInfo.routes.singleElectrifiedNames.length
|
station.generalInfo.routes.singleElectrifiedNames.length
|
||||||
}`"
|
}`"
|
||||||
>
|
>
|
||||||
@@ -163,7 +176,8 @@
|
|||||||
<span
|
<span
|
||||||
v-if="station.generalInfo.routes.singleOtherNames.length != 0"
|
v-if="station.generalInfo.routes.singleOtherNames.length != 0"
|
||||||
class="track no-catenary"
|
class="track no-catenary"
|
||||||
:title="`${$t('sceneries.info.single-track-routes-other')}${
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="`${$t('sceneries.info.single-track-routes-other')}${
|
||||||
station.generalInfo.routes.singleOtherNames.length
|
station.generalInfo.routes.singleOtherNames.length
|
||||||
}`"
|
}`"
|
||||||
>
|
>
|
||||||
@@ -177,7 +191,8 @@
|
|||||||
<span
|
<span
|
||||||
v-if="station.generalInfo.routes.doubleElectrifiedNames.length != 0"
|
v-if="station.generalInfo.routes.doubleElectrifiedNames.length != 0"
|
||||||
class="track catenary"
|
class="track catenary"
|
||||||
:title="`${$t('sceneries.info.double-track-routes-catenary')}${
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="`${$t('sceneries.info.double-track-routes-catenary')}${
|
||||||
station.generalInfo.routes.doubleElectrifiedNames.length
|
station.generalInfo.routes.doubleElectrifiedNames.length
|
||||||
}`"
|
}`"
|
||||||
>
|
>
|
||||||
@@ -187,7 +202,8 @@
|
|||||||
<span
|
<span
|
||||||
v-if="station.generalInfo.routes.doubleOtherNames.length != 0"
|
v-if="station.generalInfo.routes.doubleOtherNames.length != 0"
|
||||||
class="track no-catenary"
|
class="track no-catenary"
|
||||||
:title="`${$t('sceneries.info.double-track-routes-other')}${
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="`${$t('sceneries.info.double-track-routes-other')}${
|
||||||
station.generalInfo.routes.doubleOtherNames.length
|
station.generalInfo.routes.doubleOtherNames.length
|
||||||
}`"
|
}`"
|
||||||
>
|
>
|
||||||
@@ -201,7 +217,8 @@
|
|||||||
v-if="station.generalInfo?.signalType"
|
v-if="station.generalInfo?.signalType"
|
||||||
class="scenery-icon icon-info"
|
class="scenery-icon icon-info"
|
||||||
:class="station.generalInfo?.controlType.replace('+', '-')"
|
:class="station.generalInfo?.controlType.replace('+', '-')"
|
||||||
:title="
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="
|
||||||
$t('sceneries.info.control-type') +
|
$t('sceneries.info.control-type') +
|
||||||
$t(`controls.${station.generalInfo?.controlType}`)
|
$t(`controls.${station.generalInfo?.controlType}`)
|
||||||
"
|
"
|
||||||
@@ -214,7 +231,8 @@
|
|||||||
class="icon-info"
|
class="icon-info"
|
||||||
:src="`/images/icon-${station.generalInfo.signalType}.svg`"
|
:src="`/images/icon-${station.generalInfo.signalType}.svg`"
|
||||||
:alt="station.generalInfo.signalType"
|
:alt="station.generalInfo.signalType"
|
||||||
:title="
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="
|
||||||
$t('sceneries.info.signals-type') + $t(`signals.${station.generalInfo.signalType}`)
|
$t('sceneries.info.signals-type') + $t(`signals.${station.generalInfo.signalType}`)
|
||||||
"
|
"
|
||||||
/>
|
/>
|
||||||
@@ -224,7 +242,8 @@
|
|||||||
class="icon-info"
|
class="icon-info"
|
||||||
src="/images/icon-SUP.svg"
|
src="/images/icon-SUP.svg"
|
||||||
alt="SUP (RASP-UZK)"
|
alt="SUP (RASP-UZK)"
|
||||||
:title="$t('sceneries.info.SUP')"
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('sceneries.info.SUP')"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<img
|
<img
|
||||||
@@ -232,7 +251,8 @@
|
|||||||
class="icon-info"
|
class="icon-info"
|
||||||
src="/images/icon-ASDEK.svg"
|
src="/images/icon-ASDEK.svg"
|
||||||
alt="dSAT ASDEK"
|
alt="dSAT ASDEK"
|
||||||
:title="$t('sceneries.info.ASDEK')"
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('sceneries.info.ASDEK')"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<img
|
<img
|
||||||
@@ -240,7 +260,8 @@
|
|||||||
class="icon-info"
|
class="icon-info"
|
||||||
src="/images/icon-unknown.svg"
|
src="/images/icon-unknown.svg"
|
||||||
alt="icon-unknown"
|
alt="icon-unknown"
|
||||||
:title="$t('sceneries.info.unknown')"
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('sceneries.info.unknown')"
|
||||||
/>
|
/>
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
@@ -248,7 +269,7 @@
|
|||||||
class="station-users"
|
class="station-users"
|
||||||
:class="{ inactive: !station.onlineInfo }"
|
:class="{ inactive: !station.onlineInfo }"
|
||||||
data-tooltip-type="UsersTooltip"
|
data-tooltip-type="UsersTooltip"
|
||||||
:data-tooltip-content="JSON.stringify(station.onlineInfo?.stationTrains ?? [])"
|
:data-tooltip-content="getUsersTooltipContent(station.onlineInfo?.stationTrains ?? [])"
|
||||||
>
|
>
|
||||||
<span class="text--primary">{{
|
<span class="text--primary">{{
|
||||||
station.onlineInfo?.stationTrains?.length ?? '-'
|
station.onlineInfo?.stationTrains?.length ?? '-'
|
||||||
@@ -318,7 +339,7 @@ import dateMixin from '../../mixins/dateMixin';
|
|||||||
import styleMixin from '../../mixins/styleMixin';
|
import styleMixin from '../../mixins/styleMixin';
|
||||||
import { useApiStore } from '../../store/apiStore';
|
import { useApiStore } from '../../store/apiStore';
|
||||||
import { useMainStore } from '../../store/mainStore';
|
import { useMainStore } from '../../store/mainStore';
|
||||||
import { Station, Status } from '../../typings/common';
|
import { Station, Status, TooltipUserTrain, Train } from '../../typings/common';
|
||||||
import { useTooltipStore } from '../../store/tooltipStore';
|
import { useTooltipStore } from '../../store/tooltipStore';
|
||||||
import { getChangedFilters } from '../../managers/stationFilterManager';
|
import { getChangedFilters } from '../../managers/stationFilterManager';
|
||||||
import { ActiveSorter, HeadIdsType, headIconsIds, headIds } from './typings';
|
import { ActiveSorter, HeadIdsType, headIconsIds, headIds } from './typings';
|
||||||
@@ -394,15 +415,25 @@ export default defineComponent({
|
|||||||
else this.activeSorter.dir = 1;
|
else this.activeSorter.dir = 1;
|
||||||
|
|
||||||
this.activeSorter.headerName = headerName;
|
this.activeSorter.headerName = headerName;
|
||||||
|
},
|
||||||
|
|
||||||
|
getUsersTooltipContent(stationTrains: Train[]): string {
|
||||||
|
const usersTrains: TooltipUserTrain[] = stationTrains.map((train) => ({
|
||||||
|
driverName: train.driverName,
|
||||||
|
trainNo: train.trainNo
|
||||||
|
}));
|
||||||
|
|
||||||
|
return JSON.stringify(usersTrains);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/variables.scss';
|
@use '../../styles/icons';
|
||||||
@import '../../styles/icons.scss';
|
|
||||||
|
@use 'sass:color';
|
||||||
|
|
||||||
$rowCol: #424242;
|
$rowCol: #424242;
|
||||||
|
|
||||||
@@ -429,14 +460,16 @@ table {
|
|||||||
min-width: 1250px;
|
min-width: 1250px;
|
||||||
white-space: wrap;
|
white-space: wrap;
|
||||||
|
|
||||||
|
thead {
|
||||||
|
position: sticky;
|
||||||
|
top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
thead tr {
|
thead tr {
|
||||||
background-color: $bgCol;
|
background-color: var(--clr-bg3);
|
||||||
}
|
}
|
||||||
|
|
||||||
thead th {
|
thead th {
|
||||||
position: sticky;
|
|
||||||
top: 0;
|
|
||||||
|
|
||||||
&.station {
|
&.station {
|
||||||
width: 12em;
|
width: 12em;
|
||||||
}
|
}
|
||||||
@@ -475,7 +508,7 @@ table {
|
|||||||
}
|
}
|
||||||
|
|
||||||
padding: 0.5em 0.25em;
|
padding: 0.5em 0.25em;
|
||||||
background-color: $bgCol;
|
background-color: var(--clr-bg3);
|
||||||
white-space: pre-wrap;
|
white-space: pre-wrap;
|
||||||
|
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
@@ -502,13 +535,13 @@ tr,
|
|||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
|
|
||||||
&:nth-child(even) {
|
&:nth-child(even) {
|
||||||
background-color: lighten($rowCol, 5);
|
background-color: color.adjust($rowCol, $lightness: 5%);
|
||||||
color: white;
|
color: white;
|
||||||
}
|
}
|
||||||
|
|
||||||
&:hover,
|
&:hover,
|
||||||
&:focus {
|
&:focus {
|
||||||
background-color: lighten($rowCol, 20);
|
background-color: color.adjust($rowCol, $lightness: 15%);
|
||||||
}
|
}
|
||||||
|
|
||||||
td {
|
td {
|
||||||
@@ -522,7 +555,7 @@ tr,
|
|||||||
opacity: 0.2;
|
opacity: 0.2;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen() {
|
@include responsive.smallScreen {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
padding: 0.3em 0.5em;
|
padding: 0.3em 0.5em;
|
||||||
font-size: 1em;
|
font-size: 1em;
|
||||||
@@ -535,7 +568,7 @@ tr,
|
|||||||
max-width: 200px;
|
max-width: 200px;
|
||||||
|
|
||||||
&.default {
|
&.default {
|
||||||
color: $accentCol;
|
color: var(--clr-primary);
|
||||||
}
|
}
|
||||||
|
|
||||||
&.nonPublic {
|
&.nonPublic {
|
||||||
|
|||||||
@@ -120,6 +120,8 @@ function filterSliderValues(filters: Record<string, any>, generalInfo: StationGe
|
|||||||
const otherAvailability =
|
const otherAvailability =
|
||||||
availability == 'nonPublic' || availability == 'unavailable' || availability == 'abandoned';
|
availability == 'nonPublic' || availability == 'unavailable' || availability == 'abandoned';
|
||||||
|
|
||||||
|
const internalRoutes = routes.all.filter((r) => r.isInternal && !r.isRouteSBL && !r.hidden);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
filters['minLevel'] > reqLevel + (otherAvailability ? 1 : 0) ||
|
filters['minLevel'] > reqLevel + (otherAvailability ? 1 : 0) ||
|
||||||
filters['maxLevel'] < reqLevel + (otherAvailability ? 1 : 0) ||
|
filters['maxLevel'] < reqLevel + (otherAvailability ? 1 : 0) ||
|
||||||
@@ -130,16 +132,25 @@ function filterSliderValues(filters: Record<string, any>, generalInfo: StationGe
|
|||||||
filters['minOneWayCatenary'] > routes.singleElectrifiedNames.length ||
|
filters['minOneWayCatenary'] > routes.singleElectrifiedNames.length ||
|
||||||
filters['minOneWay'] > routes.singleOtherNames.length ||
|
filters['minOneWay'] > routes.singleOtherNames.length ||
|
||||||
filters['minTwoWayCatenary'] > routes.doubleElectrifiedNames.length ||
|
filters['minTwoWayCatenary'] > routes.doubleElectrifiedNames.length ||
|
||||||
filters['minTwoWay'] > routes.doubleOtherNames.length
|
filters['minTwoWay'] > routes.doubleOtherNames.length ||
|
||||||
|
filters['minOneWayCatenaryInt'] >
|
||||||
|
internalRoutes.filter((r) => r.routeTracks == 1 && r.isElectric == true).length ||
|
||||||
|
filters['minOneWayInt'] >
|
||||||
|
internalRoutes.filter((r) => r.routeTracks == 1 && r.isElectric == false).length ||
|
||||||
|
filters['minTwoWayCatenaryInt'] >
|
||||||
|
internalRoutes.filter((r) => r.routeTracks == 2 && r.isElectric == true).length ||
|
||||||
|
filters['minTwoWayInt'] >
|
||||||
|
internalRoutes.filter((r) => r.routeTracks == 2 && r.isElectric == false).length
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function filterInputValues(filters: Record<string, any>, generalInfo: StationGeneralInfo) {
|
function filterInputValues(filters: Record<string, any>, generalInfo: StationGeneralInfo) {
|
||||||
return (
|
return (
|
||||||
filters['authors'].length > 3 &&
|
(filters['authors'].length > 3 &&
|
||||||
!generalInfo.authors
|
!generalInfo.authors
|
||||||
?.map((a) => a.toLocaleLowerCase())
|
?.map((a) => a.toLocaleLowerCase())
|
||||||
.includes(filters['authors'].toLocaleLowerCase())
|
.includes(filters['authors'].toLocaleLowerCase())) ||
|
||||||
|
(filters['projects'].length > 0 && generalInfo.project != filters['projects'])
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="tooltip-content">
|
<div class="tooltip-content">
|
||||||
<img src="/images/icon-diamond.svg" alt="" />
|
<img src="/images/icon-diamond.svg" alt="donator diamond icon" />
|
||||||
<span>{{ tooltipStore.content }}</span>
|
<span>{{ tooltipStore.content }}</span>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@@ -20,7 +20,10 @@ export default defineComponent({
|
|||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.tooltip-content {
|
.tooltip-content {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
|
||||||
padding: 0.5em;
|
padding: 0.5em;
|
||||||
border-radius: 0.25em;
|
border-radius: 0.25em;
|
||||||
@@ -34,6 +37,5 @@ export default defineComponent({
|
|||||||
img {
|
img {
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
height: 1em;
|
height: 1em;
|
||||||
margin-right: 0.5em;
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -0,0 +1,39 @@
|
|||||||
|
<template>
|
||||||
|
<div class="tooltip-content">
|
||||||
|
<span v-html="tooltipStore.content"></span>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent } from 'vue';
|
||||||
|
import { useTooltipStore } from '../../store/tooltipStore';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
tooltipStore: useTooltipStore()
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.tooltip-content {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.5em;
|
||||||
|
white-space: pre-line;
|
||||||
|
|
||||||
|
padding: 0.25em 0.5em;
|
||||||
|
border-radius: 0.25em;
|
||||||
|
|
||||||
|
width: 100%;
|
||||||
|
background-color: #1f1f1f;
|
||||||
|
box-shadow: 0 0 5px 2px #aaa;
|
||||||
|
}
|
||||||
|
|
||||||
|
img {
|
||||||
|
height: 1em;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -32,12 +32,11 @@ export default defineComponent({
|
|||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.tooltip-content {
|
.tooltip-content {
|
||||||
width: 300px;
|
width: 200px;
|
||||||
|
|
||||||
padding: 0.25em 0.5em;
|
padding: 0.25em 0.5em;
|
||||||
border-radius: 0.25em;
|
border-radius: 0.25em;
|
||||||
|
|
||||||
width: 100%;
|
|
||||||
background-color: #1b1b1b;
|
background-color: #1b1b1b;
|
||||||
box-shadow: 0 0 5px 2px #aaa;
|
box-shadow: 0 0 5px 2px #aaa;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,11 +12,21 @@ import VehiclePreviewTooltip from './VehiclePreviewTooltip.vue';
|
|||||||
import BaseTooltip from './BaseTooltip.vue';
|
import BaseTooltip from './BaseTooltip.vue';
|
||||||
import SpawnsTooltip from './SpawnsTooltip.vue';
|
import SpawnsTooltip from './SpawnsTooltip.vue';
|
||||||
import UsersTooltip from './UsersTooltip.vue';
|
import UsersTooltip from './UsersTooltip.vue';
|
||||||
|
import HtmlTooltip from './HtmlTooltip.vue';
|
||||||
|
import TrainInfoTooltip from "./TrainInfoTooltip.vue";
|
||||||
|
|
||||||
const BOX_PADDING_PX = 20;
|
const BOX_PADDING_PX = 20;
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: { DonatorTooltip, VehiclePreviewTooltip, BaseTooltip, SpawnsTooltip, UsersTooltip },
|
components: {
|
||||||
|
DonatorTooltip,
|
||||||
|
VehiclePreviewTooltip,
|
||||||
|
BaseTooltip,
|
||||||
|
SpawnsTooltip,
|
||||||
|
UsersTooltip,
|
||||||
|
HtmlTooltip,
|
||||||
|
TrainInfoTooltip
|
||||||
|
},
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -0,0 +1,78 @@
|
|||||||
|
<template>
|
||||||
|
<div class="tooltip-content">
|
||||||
|
<span v-if="trainInfo">
|
||||||
|
<b v-if="trainInfo.timetableData" style="text-transform: uppercase">
|
||||||
|
<span class="text--primary">{{ trainInfo.timetableData.category }}</span>
|
||||||
|
{{ getCategoryExplanation(trainInfo.timetableData.category) }}
|
||||||
|
</b>
|
||||||
|
|
||||||
|
<div class="text--primary">
|
||||||
|
<b>{{ trainInfo.stockList[0] }}</b> • {{ trainInfo.length }}m •
|
||||||
|
{{ (trainInfo.mass / 1000).toFixed(2) }}t
|
||||||
|
<span v-if="trainInfo.timetableData">
|
||||||
|
• vRJ:
|
||||||
|
{{
|
||||||
|
trainInfo.timetableData?.trainMaxSpeed ||
|
||||||
|
getStockSpeedLimit(trainInfo.stockList, trainInfo.mass)
|
||||||
|
}}km/h
|
||||||
|
</span>
|
||||||
|
<span v-else class="text--grayed font--italic">
|
||||||
|
• vMax:
|
||||||
|
{{ getStockSpeedLimit(trainInfo.stockList, trainInfo.mass) }}km/h
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="text--grayed">
|
||||||
|
{{ displayTrainPosition(trainInfo) }} - {{ trainInfo.speed }}km/h
|
||||||
|
<span v-if="!trainInfo.online" style="color: salmon">
|
||||||
|
- offline {{ lastSeenMessage(trainInfo.lastSeen) }}</span
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div></div>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { defineComponent } from 'vue';
|
||||||
|
import { useTooltipStore } from '../../store/tooltipStore';
|
||||||
|
import trainCategoryMixin from '../../mixins/trainCategoryMixin';
|
||||||
|
import trainInfoMixin from '../../mixins/trainInfoMixin';
|
||||||
|
import { useMainStore } from '../../store/mainStore';
|
||||||
|
|
||||||
|
export default defineComponent({
|
||||||
|
mixins: [trainCategoryMixin, trainInfoMixin],
|
||||||
|
|
||||||
|
data: () => ({
|
||||||
|
tooltipStore: useTooltipStore(),
|
||||||
|
mainStore: useMainStore()
|
||||||
|
}),
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
trainInfo() {
|
||||||
|
if (this.tooltipStore.content == '') return null;
|
||||||
|
|
||||||
|
// Passed "content" string should be the desired train's ID
|
||||||
|
return this.mainStore.trainList.find((t) => t.id === this.tooltipStore.content);
|
||||||
|
},
|
||||||
|
|
||||||
|
lastSceneryStatus() {}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.tooltip-content {
|
||||||
|
padding: 0.25em 0.5em;
|
||||||
|
border-radius: 0.25em;
|
||||||
|
|
||||||
|
width: 100%;
|
||||||
|
background-color: #1f1f1f;
|
||||||
|
box-shadow: 0 0 5px 2px #aaa;
|
||||||
|
}
|
||||||
|
|
||||||
|
img {
|
||||||
|
height: 1em;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -10,7 +10,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent } from 'vue';
|
import { defineComponent } from 'vue';
|
||||||
import { useTooltipStore } from '../../store/tooltipStore';
|
import { useTooltipStore } from '../../store/tooltipStore';
|
||||||
import { Train } from '../../typings/common';
|
import { TooltipUserTrain } from '../../typings/common';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
data() {
|
data() {
|
||||||
@@ -23,7 +23,7 @@ export default defineComponent({
|
|||||||
trains() {
|
trains() {
|
||||||
if (this.tooltipStore.content == '') return [];
|
if (this.tooltipStore.content == '') return [];
|
||||||
|
|
||||||
const parsedTrains = JSON.parse(this.tooltipStore.content) as Train[];
|
const parsedTrains = JSON.parse(this.tooltipStore.content) as TooltipUserTrain[];
|
||||||
return (parsedTrains ?? []).sort((a, b) => a.trainNo - b.trainNo);
|
return (parsedTrains ?? []).sort((a, b) => a.trainNo - b.trainNo);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -32,12 +32,11 @@ export default defineComponent({
|
|||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
.tooltip-content {
|
.tooltip-content {
|
||||||
width: 300px;
|
width: 200px;
|
||||||
|
|
||||||
padding: 0.25em 0.5em;
|
padding: 0.25em 0.5em;
|
||||||
border-radius: 0.25em;
|
border-radius: 0.25em;
|
||||||
|
|
||||||
width: 100%;
|
|
||||||
background-color: #1b1b1b;
|
background-color: #1b1b1b;
|
||||||
box-shadow: 0 0 5px 2px #aaa;
|
box-shadow: 0 0 5px 2px #aaa;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="tooltip-content">
|
<div class="tooltip-content">
|
||||||
<div v-if="imageState == 'loading'" class="loading-info">
|
<div class="image-box">
|
||||||
{{ $t('vehicle-preview.loading') }}
|
<Loading v-if="imageState == 'loading'" class="loading-info" />
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-if="imageState == 'error'">{{ $t('vehicle-preview.error') }}</div>
|
|
||||||
|
|
||||||
<img
|
<img
|
||||||
v-if="tooltipStore.type"
|
v-if="tooltipStore.type"
|
||||||
@@ -12,11 +9,9 @@
|
|||||||
@error="onImageError"
|
@error="onImageError"
|
||||||
width="300"
|
width="300"
|
||||||
height="176"
|
height="176"
|
||||||
class="rounded-md w-full h-auto"
|
:src="`https://stacjownik.spythere.eu/static/images/${vehicleName}--300px.jpg`"
|
||||||
:src="`https://static.spythere.eu/images/${vehicleName}--300px.jpg`"
|
|
||||||
/>
|
/>
|
||||||
|
</div>
|
||||||
<div v-if="imageState == 'error'" class="error-placeholder"></div>
|
|
||||||
|
|
||||||
<div class="vehicle-name">
|
<div class="vehicle-name">
|
||||||
{{ vehicleName.replace(/_/g, ' ') }}
|
{{ vehicleName.replace(/_/g, ' ') }}
|
||||||
@@ -35,8 +30,11 @@
|
|||||||
import { defineComponent } from 'vue';
|
import { defineComponent } from 'vue';
|
||||||
import { useTooltipStore } from '../../store/tooltipStore';
|
import { useTooltipStore } from '../../store/tooltipStore';
|
||||||
import { useApiStore } from '../../store/apiStore';
|
import { useApiStore } from '../../store/apiStore';
|
||||||
|
import Loading from '../Global/Loading.vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
|
components: { Loading },
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
tooltipStore: useTooltipStore(),
|
tooltipStore: useTooltipStore(),
|
||||||
@@ -61,9 +59,12 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
onImageError(e: Event) {
|
onImageError(e: Event) {
|
||||||
|
if (!e.target || !(e.target instanceof HTMLImageElement) || this.imageState == 'error')
|
||||||
|
return;
|
||||||
|
|
||||||
this.imageState = 'error';
|
this.imageState = 'error';
|
||||||
|
|
||||||
(e.target as HTMLElement).style.display = 'none';
|
e.target.src = '/images/no-vehicle-image.png';
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -77,9 +78,11 @@ export default defineComponent({
|
|||||||
},
|
},
|
||||||
|
|
||||||
vehicleCargo() {
|
vehicleCargo() {
|
||||||
return this.vehicleData?.group.cargoTypes?.find(
|
const x = this.vehicleData?.group.cargoTypes?.find(
|
||||||
(c) => c.id == this.tooltipStore.content.split(':')[1]
|
(c) => c.id == this.tooltipStore.content.split(':')[1]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
return x;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -96,10 +99,16 @@ export default defineComponent({
|
|||||||
border-radius: 0.5em;
|
border-radius: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.image-box {
|
||||||
|
position: relative;
|
||||||
|
min-height: 170px;
|
||||||
|
}
|
||||||
|
|
||||||
.loading-info {
|
.loading-info {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
left: 50%;
|
left: 50%;
|
||||||
transform: translateX(-50%);
|
top: 35%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
}
|
}
|
||||||
|
|
||||||
img {
|
img {
|
||||||
|
|||||||
@@ -4,10 +4,12 @@
|
|||||||
:data-minor="stop.isSBL || (stop.nameRaw.endsWith(', po') && !stop.duration)"
|
:data-minor="stop.isSBL || (stop.nameRaw.endsWith(', po') && !stop.duration)"
|
||||||
>
|
>
|
||||||
<router-link v-if="/(, podg$|<strong>)/.test(stop.nameHtml)" :to="sceneryHref">
|
<router-link v-if="/(, podg$|<strong>)/.test(stop.nameHtml)" :to="sceneryHref">
|
||||||
<span class="stop-name" v-html="stop.nameHtml"></span>
|
<b class="stop-name">
|
||||||
|
{{ stop.nameRaw }}
|
||||||
|
</b>
|
||||||
</router-link>
|
</router-link>
|
||||||
|
|
||||||
<span v-else class="stop-name" v-html="stop.nameHtml"></span>
|
<span v-else class="stop-name">{{ stop.nameRaw }}</span>
|
||||||
|
|
||||||
<span
|
<span
|
||||||
v-if="stop.position != 'begin'"
|
v-if="stop.position != 'begin'"
|
||||||
@@ -45,7 +47,10 @@
|
|||||||
<span
|
<span
|
||||||
v-if="
|
v-if="
|
||||||
stop.position != 'end' &&
|
stop.position != 'end' &&
|
||||||
(stop.duration != 0 || stop.status == 'stopped' || stop.departureDelay != stop.arrivalDelay)
|
(stop.duration != 0 ||
|
||||||
|
stop.position == 'begin' ||
|
||||||
|
stop.status == 'stopped' ||
|
||||||
|
stop.departureDelay != stop.arrivalDelay)
|
||||||
"
|
"
|
||||||
class="date departure"
|
class="date departure"
|
||||||
:data-status-delayed="stop.departureDelay > 0"
|
:data-status-delayed="stop.departureDelay > 0"
|
||||||
@@ -68,16 +73,16 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { PropType, defineComponent } from 'vue';
|
import { PropType, defineComponent, stop } from 'vue';
|
||||||
import dateMixin from '../../mixins/dateMixin';
|
import dateMixin from '../../mixins/dateMixin';
|
||||||
import { TrainScheduleStop } from './TrainSchedule.vue';
|
import { TrainSchedulePoint } from './typings';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
mixins: [dateMixin],
|
mixins: [dateMixin],
|
||||||
|
|
||||||
props: {
|
props: {
|
||||||
stop: {
|
stop: {
|
||||||
type: Object as PropType<TrainScheduleStop>,
|
type: Object as PropType<TrainSchedulePoint>,
|
||||||
required: true
|
required: true
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -139,6 +144,10 @@ s {
|
|||||||
color: #aaa;
|
color: #aaa;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
i {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.stop {
|
.stop {
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="train-info" :data-extended="extended">
|
<div class="train-info" :data-extended="extended">
|
||||||
<section class="train-general">
|
|
||||||
<div class="general-top-bar">
|
<div class="general-top-bar">
|
||||||
<div class="top-bar-header">
|
<div class="top-bar-header">
|
||||||
<b class="warning-timeout" v-if="train.isTimeout" :title="$t('trains.timeout')">?</b>
|
<b class="warning-timeout" v-if="train.isTimeout" :title="$t('trains.timeout')">?</b>
|
||||||
@@ -10,20 +9,29 @@
|
|||||||
|
|
||||||
<span
|
<span
|
||||||
class="train-badge twr"
|
class="train-badge twr"
|
||||||
v-if="train.timetableData?.TWR"
|
v-if="train.timetableData?.twr"
|
||||||
data-tooltip-type="BaseTooltip"
|
data-tooltip-type="BaseTooltip"
|
||||||
:data-tooltip-content="$t('general.TWR') + `:\n${train.timetableData.warningNotes}`"
|
:data-tooltip-content="$t('warnings.TWR')"
|
||||||
>
|
>
|
||||||
TWR
|
TWR
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<span
|
<span
|
||||||
class="train-badge skr"
|
class="train-badge tn"
|
||||||
v-if="train.timetableData?.SKR"
|
v-if="train.timetableData?.hasDangerousCargo"
|
||||||
data-tooltip-type="BaseTooltip"
|
data-tooltip-type="BaseTooltip"
|
||||||
:data-tooltip-content="$t('general.SKR')"
|
:data-tooltip-content="$t('warnings.TN')"
|
||||||
>
|
>
|
||||||
SKR
|
TN
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span
|
||||||
|
class="train-badge pn"
|
||||||
|
v-if="train.timetableData?.hasExtraDeliveries"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('warnings.PN')"
|
||||||
|
>
|
||||||
|
PN
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<b
|
<b
|
||||||
@@ -34,7 +42,9 @@
|
|||||||
>
|
>
|
||||||
{{ train.timetableData.category }}
|
{{ train.timetableData.category }}
|
||||||
</b>
|
</b>
|
||||||
|
|
||||||
<b class="train-number">{{ train.trainNo }}</b>
|
<b class="train-number">{{ train.trainNo }}</b>
|
||||||
|
|
||||||
<span>•</span>
|
<span>•</span>
|
||||||
|
|
||||||
<div class="train-driver">
|
<div class="train-driver">
|
||||||
@@ -50,7 +60,7 @@
|
|||||||
data-tooltip-type="DonatorTooltip"
|
data-tooltip-type="DonatorTooltip"
|
||||||
:data-tooltip-content="$t('donations.driver-message')"
|
:data-tooltip-content="$t('donations.driver-message')"
|
||||||
>
|
>
|
||||||
{{ train.driverName }}
|
<span class="text--donator">{{ train.driverName }} </span>
|
||||||
<img src="/images/icon-diamond.svg" alt="donator diamond icon" />
|
<img src="/images/icon-diamond.svg" alt="donator diamond icon" />
|
||||||
</b>
|
</b>
|
||||||
|
|
||||||
@@ -96,12 +106,15 @@
|
|||||||
|
|
||||||
<div class="status-badges">
|
<div class="status-badges">
|
||||||
<div v-if="!train.currentStationHash" class="train-badge offline">
|
<div v-if="!train.currentStationHash" class="train-badge offline">
|
||||||
<img src="/images/icon-offline.svg" alt="offline train icon" />
|
<i class="fa-solid fa-ban"></i>
|
||||||
{{ $t('trains.scenery-offline') }}
|
{{ $t('trains.scenery-offline') }}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="!train.online" class="train-badge offline">
|
<div
|
||||||
<img src="/images/icon-offline.svg" alt="offline train icon" />
|
v-if="!train.online && train.lastSeen <= Date.now() - 60000"
|
||||||
|
class="train-badge offline"
|
||||||
|
>
|
||||||
|
<i class="fa-solid fa-user-slash"></i>
|
||||||
Offline {{ lastSeenMessage(train.lastSeen) }}
|
Offline {{ lastSeenMessage(train.lastSeen) }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -122,7 +135,11 @@
|
|||||||
<img src="/images/icon-speed.svg" alt="speed icon" />
|
<img src="/images/icon-speed.svg" alt="speed icon" />
|
||||||
{{ train.speed }} km/h
|
{{ train.speed }} km/h
|
||||||
|
|
||||||
<span v-if="stockSpeedLimit != Infinity">
|
<span v-if="train.timetableData">
|
||||||
|
• vRJ: {{ train.timetableData.trainMaxSpeed }} km/h
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<span v-else-if="stockSpeedLimit != Infinity">
|
||||||
•
|
•
|
||||||
<em
|
<em
|
||||||
class="text--grayed"
|
class="text--grayed"
|
||||||
@@ -130,6 +147,7 @@
|
|||||||
tabindex="0"
|
tabindex="0"
|
||||||
:data-tooltip="$t('trains.vmax-tooltip')"
|
:data-tooltip="$t('trains.vmax-tooltip')"
|
||||||
>
|
>
|
||||||
|
vMax:
|
||||||
{{ stockSpeedLimit }} km/h
|
{{ stockSpeedLimit }} km/h
|
||||||
</em>
|
</em>
|
||||||
</span>
|
</span>
|
||||||
@@ -142,49 +160,45 @@
|
|||||||
|
|
||||||
<div
|
<div
|
||||||
class="train-dangers"
|
class="train-dangers"
|
||||||
v-if="extended && (train.timetableData?.TWR || train.timetableData?.SKR)"
|
v-if="extended && train.timetableData && train.timetableData.warningNotes"
|
||||||
>
|
>
|
||||||
<div v-if="train.timetableData.TWR">
|
<div class="dangers-badges">
|
||||||
<b style="color: var(--clr-twr)">TWR</b> - {{ $t('general.TWR') }}
|
<div v-if="train.timetableData?.twr">
|
||||||
<i>({{ train.timetableData?.warningNotes }})</i>
|
<div class="train-badge twr">TWR</div>
|
||||||
|
- {{ $t('warnings.TWR') }}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="train.timetableData.SKR">
|
<div v-if="train.timetableData?.hasDangerousCargo">
|
||||||
<b style="color: var(--clr-skr)">SKR</b> - {{ $t('general.SKR') }}
|
<div class="train-badge tn">TN</div>
|
||||||
|
- {{ $t('warnings.TN') }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<section class="train-stats" v-if="!extended">
|
<div v-if="train.timetableData?.hasExtraDeliveries">
|
||||||
<StockList :trainStockList="train.stockList" :tractionOnly="true" />
|
<div class="train-badge pn">PN</div>
|
||||||
|
- {{ $t('warnings.PN') }}
|
||||||
<div>
|
</div>
|
||||||
<span>{{ train.speed }}km/h</span>
|
</div>
|
||||||
|
|
||||||
<div>
|
<div class="dangers-notes">
|
||||||
<span> {{ train.length }}m</span>
|
<h4>{{ $t('warnings.header-title') }}</h4>
|
||||||
•
|
<p>
|
||||||
<span> {{ (train.mass / 1000).toFixed(1) }}t</span>
|
<i>{{ train.timetableData?.warningNotes }}</i>
|
||||||
<span v-if="train.stockList.length > 1">
|
</p>
|
||||||
•
|
|
||||||
{{ $t('trains.cars') }}: {{ train.stockList.length - 1 }}
|
|
||||||
</span>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent } from 'vue';
|
import { defineComponent } from 'vue';
|
||||||
import styleMixin from '../../mixins/styleMixin';
|
|
||||||
import trainInfoMixin from '../../mixins/trainInfoMixin';
|
|
||||||
import ProgressBar from '../Global/ProgressBar.vue';
|
|
||||||
import { useMainStore } from '../../store/mainStore';
|
import { useMainStore } from '../../store/mainStore';
|
||||||
import { useApiStore } from '../../store/apiStore';
|
import { useApiStore } from '../../store/apiStore';
|
||||||
import StockList from '../Global/StockList.vue';
|
|
||||||
import { Train } from '../../typings/common';
|
import { Train } from '../../typings/common';
|
||||||
|
import styleMixin from '../../mixins/styleMixin';
|
||||||
|
import trainInfoMixin from '../../mixins/trainInfoMixin';
|
||||||
import trainCategoryMixin from '../../mixins/trainCategoryMixin';
|
import trainCategoryMixin from '../../mixins/trainCategoryMixin';
|
||||||
|
import ProgressBar from '../Global/ProgressBar.vue';
|
||||||
|
import StockList from '../Global/StockList.vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
mixins: [trainInfoMixin, styleMixin, trainCategoryMixin],
|
mixins: [trainInfoMixin, styleMixin, trainCategoryMixin],
|
||||||
@@ -209,14 +223,9 @@ export default defineComponent({
|
|||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
stockSpeedLimit() {
|
stockSpeedLimit() {
|
||||||
return this.train.stockList.reduce((acc, stockName) => {
|
return this.getStockSpeedLimit(this.train.stockList, this.train.mass);
|
||||||
const vehicleSpeed =
|
|
||||||
this.apiStore.vehiclesData?.find((v) => v.name == stockName.split(':')[0])?.group.speed ??
|
|
||||||
300;
|
|
||||||
|
|
||||||
return Math.min(vehicleSpeed, acc);
|
|
||||||
}, 300);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
journalRouteLocation() {
|
journalRouteLocation() {
|
||||||
return {
|
return {
|
||||||
path: '/journal/timetables',
|
path: '/journal/timetables',
|
||||||
@@ -230,8 +239,7 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/badge';
|
||||||
@import '../../styles/badge.scss';
|
|
||||||
|
|
||||||
.image-warning {
|
.image-warning {
|
||||||
height: 1em;
|
height: 1em;
|
||||||
@@ -239,31 +247,33 @@ export default defineComponent({
|
|||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
|
||||||
.train-stats {
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
flex-direction: column;
|
|
||||||
text-align: center;
|
|
||||||
|
|
||||||
line-height: 1.5em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.train-dangers {
|
.train-dangers {
|
||||||
margin-top: 0.5em;
|
margin-top: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.train-info {
|
.dangers-badges {
|
||||||
display: grid;
|
display: flex;
|
||||||
grid-template-columns: 2fr 1fr;
|
flex-direction: column;
|
||||||
grid-template-rows: 1fr;
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
&[data-extended='true'] {
|
.dangers-notes {
|
||||||
grid-template-columns: 1fr;
|
margin-top: 0.5em;
|
||||||
|
white-space: pre-wrap;
|
||||||
|
|
||||||
|
p {
|
||||||
|
margin-top: 0.25em;
|
||||||
|
max-height: 200px;
|
||||||
|
max-width: 500px;
|
||||||
|
overflow: auto;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
padding: 1em;
|
.train-info {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
font-size: 1em;
|
||||||
|
gap: 0.25em;
|
||||||
|
|
||||||
background-color: #1a1a1a;
|
background-color: #1a1a1a;
|
||||||
gap: 0.5em;
|
gap: 0.5em;
|
||||||
@@ -293,12 +303,6 @@ export default defineComponent({
|
|||||||
padding: 0 0.25em;
|
padding: 0 0.25em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.train-general {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 0.25em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.general-stops {
|
.general-stops {
|
||||||
font-size: 0.8em;
|
font-size: 0.8em;
|
||||||
}
|
}
|
||||||
@@ -349,6 +353,7 @@ export default defineComponent({
|
|||||||
.status-badges {
|
.status-badges {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
|
margin-left: 0.25em;
|
||||||
|
|
||||||
gap: 0.25em;
|
gap: 0.25em;
|
||||||
|
|
||||||
@@ -375,11 +380,4 @@ export default defineComponent({
|
|||||||
display: flex;
|
display: flex;
|
||||||
gap: 0.25em;
|
gap: 0.25em;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen() {
|
|
||||||
.train-info {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
gap: 1em 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -24,31 +24,31 @@
|
|||||||
@blur="preventKeyDown = false"
|
@blur="preventKeyDown = false"
|
||||||
:placeholder="$t(`options.search-train`)"
|
:placeholder="$t(`options.search-train`)"
|
||||||
/>
|
/>
|
||||||
<button class="search-exit">
|
<button class="btn btn--action search-exit" @click="onInputClear('train')">
|
||||||
<img
|
<img src="/images/icon-exit.svg" alt="Trains search clear icon" />
|
||||||
src="/images/icon-exit.svg"
|
|
||||||
alt="Trains search clear icon"
|
|
||||||
@click="onInputClear('train')"
|
|
||||||
/>
|
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="search-box">
|
<div class="search-box">
|
||||||
|
<datalist id="search-active-driver">
|
||||||
|
<option v-for="driverName in activeDriverNames" :value="driverName">
|
||||||
|
{{ driverName }}
|
||||||
|
</option>
|
||||||
|
</datalist>
|
||||||
|
|
||||||
<input
|
<input
|
||||||
v-model="searchedDriver"
|
|
||||||
class="search-input"
|
class="search-input"
|
||||||
id="driver-search"
|
list="search-active-driver"
|
||||||
name="driver-search"
|
name="search-active-driver"
|
||||||
|
id="search-active-driver"
|
||||||
|
:placeholder="$t(`options.search-driver`)"
|
||||||
|
v-model="searchedDriver"
|
||||||
@focus="preventKeyDown = true"
|
@focus="preventKeyDown = true"
|
||||||
@blur="preventKeyDown = false"
|
@blur="preventKeyDown = false"
|
||||||
:placeholder="$t(`options.search-driver`)"
|
|
||||||
/>
|
|
||||||
<button class="search-exit">
|
|
||||||
<img
|
|
||||||
src="/images/icon-exit.svg"
|
|
||||||
alt="Driver search clear icon"
|
|
||||||
@click="onInputClear('driver')"
|
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
<button class="btn btn--action search-exit" @click="onInputClear('driver')">
|
||||||
|
<img src="/images/icon-exit.svg" alt="Trains search clear icon" />
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -101,6 +101,7 @@
|
|||||||
import { defineComponent, inject, PropType } from 'vue';
|
import { defineComponent, inject, PropType } from 'vue';
|
||||||
import keyMixin from '../../mixins/keyMixin';
|
import keyMixin from '../../mixins/keyMixin';
|
||||||
import { TrainFilter, TrainFilterSection } from './typings';
|
import { TrainFilter, TrainFilterSection } from './typings';
|
||||||
|
import { useMainStore } from '../../store/mainStore';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
mixins: [keyMixin],
|
mixins: [keyMixin],
|
||||||
@@ -120,6 +121,7 @@ export default defineComponent({
|
|||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
showOptions: false,
|
showOptions: false,
|
||||||
|
store: useMainStore(),
|
||||||
lastSelectedFilter: null as TrainFilter | null,
|
lastSelectedFilter: null as TrainFilter | null,
|
||||||
TrainFilterSection
|
TrainFilterSection
|
||||||
};
|
};
|
||||||
@@ -141,6 +143,16 @@ export default defineComponent({
|
|||||||
id,
|
id,
|
||||||
value: this.$t(`options.sort-${id}`)
|
value: this.$t(`options.sort-${id}`)
|
||||||
}));
|
}));
|
||||||
|
},
|
||||||
|
|
||||||
|
activeDriverNames() {
|
||||||
|
const driverNameSet = new Set<string>();
|
||||||
|
|
||||||
|
this.store.trainList.forEach((train) => {
|
||||||
|
driverNameSet.add(train.driverName);
|
||||||
|
});
|
||||||
|
|
||||||
|
return [...driverNameSet].sort((a, b) => a.localeCompare(b));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -178,6 +190,12 @@ export default defineComponent({
|
|||||||
this.trainFilterList.forEach((filter) => {
|
this.trainFilterList.forEach((filter) => {
|
||||||
filter.isActive = true;
|
filter.isActive = true;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.sorterActive.id = 'routeDistance';
|
||||||
|
this.sorterActive.dir = -1;
|
||||||
|
|
||||||
|
this.searchedDriver = '';
|
||||||
|
this.searchedTrain = '';
|
||||||
},
|
},
|
||||||
|
|
||||||
onInputClear(id: 'driver' | 'train') {
|
onInputClear(id: 'driver' | 'train') {
|
||||||
@@ -189,8 +207,8 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/dropdown.scss';
|
@use '../../styles/dropdown';
|
||||||
@import '../../styles/dropdown_filters.scss';
|
@use '../../styles/dropdown-filters';
|
||||||
|
|
||||||
.search_content > div {
|
.search_content > div {
|
||||||
margin: 0.5em auto;
|
margin: 0.5em auto;
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="train-schedule" @click="toggleShowState">
|
<div class="train-schedule">
|
||||||
<StockList :trainStockList="train.stockList" />
|
|
||||||
|
|
||||||
<div class="schedule-wrapper" v-if="train.timetableData">
|
<div class="schedule-wrapper" v-if="train.timetableData">
|
||||||
<div class="stops">
|
<div class="stops">
|
||||||
<div
|
<div
|
||||||
@@ -9,11 +7,21 @@
|
|||||||
:key="i"
|
:key="i"
|
||||||
class="stop"
|
class="stop"
|
||||||
:data-status="stop.status"
|
:data-status="stop.status"
|
||||||
|
:data-sbl="stop.isSBL && stop.sceneryName == scheduleStops[i + 1]?.sceneryName"
|
||||||
:data-position="stop.position"
|
:data-position="stop.position"
|
||||||
:data-delayed="stop.departureDelay > 0"
|
:data-delayed="stop.departureDelay > 0"
|
||||||
:data-stop-type="stop.type"
|
:data-stop-type="stop.type"
|
||||||
:data-minor-stop-active="stop.isActive"
|
:data-is-active="stop.isActive"
|
||||||
:data-last-confirmed="stop.isLastConfirmed"
|
:data-track-count-departure="
|
||||||
|
stop.departureLineInfo?.routeTracks ??
|
||||||
|
stop.nextPointRef?.arrivalLineInfo?.routeTracks ??
|
||||||
|
2
|
||||||
|
"
|
||||||
|
:data-track-count-arrival="
|
||||||
|
stop.arrivalLineInfo?.routeTracks ??
|
||||||
|
scheduleStops[i - 1]?.departureLineInfo?.routeTracks ??
|
||||||
|
2
|
||||||
|
"
|
||||||
>
|
>
|
||||||
<span class="stop_info">
|
<span class="stop_info">
|
||||||
<span class="distance">
|
<span class="distance">
|
||||||
@@ -44,47 +52,108 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Routes -->
|
<!-- Routes -->
|
||||||
|
|
||||||
<span
|
<span
|
||||||
v-if="
|
v-if="
|
||||||
stop.departureLine &&
|
stop.departureLine &&
|
||||||
scheduleStops[i + 1] != undefined &&
|
(scheduleStops[i + 1]?.arrivalLineInfo?.routeSpeed !=
|
||||||
!/-|_|(^it\d+)|(^sbl)/gi.test(stop.departureLine)
|
stop.arrivalLineInfo?.routeSpeed ||
|
||||||
|
stop.sceneryName != scheduleStops[i + 1]?.sceneryName)
|
||||||
"
|
"
|
||||||
>
|
>
|
||||||
<div class="scenery-route">
|
<div class="scenery-route">
|
||||||
<span>{{ stop.departureLine }}</span>
|
<span>{{ stop.departureLine }}</span>
|
||||||
|
|
||||||
<span v-if="stop.departureLineInfo">
|
<span v-if="stop.departureLineInfo">
|
||||||
| {{ stop.departureLineInfo.routeSpeed }}
|
<span>
|
||||||
<span v-if="stop.departureLineInfo.isElectric">⚡</span>
|
|
|
||||||
|
{{
|
||||||
|
stop.departureLineInfo.routeSpeedExit &&
|
||||||
|
stop.departureLineInfo.routeSpeedExit != stop.departureLineInfo.routeSpeed
|
||||||
|
? `${stop.departureLineInfo.routeSpeedExit} (${stop.departureLineInfo.routeSpeed})`
|
||||||
|
: stop.departureLineInfo.routeSpeed
|
||||||
|
}}</span
|
||||||
|
>
|
||||||
|
|
||||||
<img
|
<img
|
||||||
v-else
|
:src="
|
||||||
src="/images/icon-we4a.png"
|
stop.departureLineInfo.isElectric
|
||||||
:title="$t('trains.we4a-tooltip')"
|
? '/images/icon-catenary.svg'
|
||||||
width="10"
|
: '/images/icon-we4a.png'
|
||||||
|
"
|
||||||
|
width="14"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="
|
||||||
|
$t(
|
||||||
|
`trains.${!stop.departureLineInfo.isElectric ? 'no-' : ''}catenary-tooltip`
|
||||||
|
)
|
||||||
|
"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<img
|
||||||
|
v-if="stop.departureLineInfo.isRouteSBL"
|
||||||
|
src="/images/icon-sbl-transparent.svg"
|
||||||
|
width="14"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('trains.sbl-tooltip')"
|
||||||
/>
|
/>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div
|
<div
|
||||||
v-if="stop.sceneryName != scheduleStops[i + 1]?.sceneryName"
|
v-if="stop.nextPointRef && stop.sceneryName != stop.nextPointRef.sceneryName"
|
||||||
class="scenery-change-name"
|
class="scenery-change-name"
|
||||||
>
|
>
|
||||||
<span>{{ scheduleStops[i + 1].sceneryName }}</span>
|
<span>{{ stop.nextPointRef.sceneryName }}</span>
|
||||||
<span v-if="stop.departureLineInfo?.routeTracks == 1"> ↕</span>
|
|
||||||
<span v-else> ⇅</span>
|
<i
|
||||||
|
v-if="!stop.nextPointRef.isSceneryOnline"
|
||||||
|
class="fa-solid fa-ban fa-sm"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('app.tooltip-scenery-offline')"
|
||||||
|
style="color: salmon; margin-left: 0.25em"
|
||||||
|
></i>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="scenery-route">
|
<div
|
||||||
<span> {{ scheduleStops[i + 1].arrivalLine }}</span>
|
class="scenery-route"
|
||||||
|
v-if="stop.nextPointRef && stop.sceneryName != stop.nextPointRef.sceneryName"
|
||||||
|
>
|
||||||
|
<span> {{ stop.nextPointRef.arrivalLine }}</span>
|
||||||
|
|
||||||
|
<span v-if="stop.nextPointRef.arrivalLineInfo">
|
||||||
|
<span> | {{ stop.nextPointRef.arrivalLineInfo.routeSpeed }}</span>
|
||||||
|
<span
|
||||||
|
v-if="
|
||||||
|
stop.nextPointRef.arrivalLineInfo.routeSpeedExit &&
|
||||||
|
stop.nextPointRef.arrivalLineInfo.routeSpeedExit !=
|
||||||
|
stop.nextPointRef.arrivalLineInfo.routeSpeed
|
||||||
|
"
|
||||||
|
>
|
||||||
|
({{ stop.nextPointRef.arrivalLineInfo.routeSpeedExit }})
|
||||||
|
</span>
|
||||||
|
|
||||||
<span v-if="scheduleStops[i + 1].arrivalLineInfo">
|
|
||||||
| {{ scheduleStops[i + 1].arrivalLineInfo!.routeSpeed }}
|
|
||||||
<span v-if="scheduleStops[i + 1].arrivalLineInfo!.isElectric">⚡</span>
|
|
||||||
<img
|
<img
|
||||||
v-else
|
:src="
|
||||||
src="/images/icon-we4a.png"
|
stop.nextPointRef.arrivalLineInfo?.isElectric
|
||||||
:title="$t('trains.we4a-tooltip')"
|
? '/images/icon-catenary.svg'
|
||||||
width="10"
|
: '/images/icon-we4a.png'
|
||||||
|
"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="
|
||||||
|
$t(
|
||||||
|
`trains.${!stop.nextPointRef.arrivalLineInfo?.isElectric ? 'no-' : ''}catenary-tooltip`
|
||||||
|
)
|
||||||
|
"
|
||||||
|
width="14"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<img
|
||||||
|
v-if="stop.nextPointRef.arrivalLineInfo!.isRouteSBL"
|
||||||
|
src="/images/icon-sbl-transparent.svg"
|
||||||
|
width="14"
|
||||||
|
data-tooltip-type="BaseTooltip"
|
||||||
|
:data-tooltip-content="$t('trains.sbl-tooltip')"
|
||||||
/>
|
/>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -104,44 +173,8 @@ import StopLabel from './StopLabel.vue';
|
|||||||
import StockList from '../Global/StockList.vue';
|
import StockList from '../Global/StockList.vue';
|
||||||
import { useMainStore } from '../../store/mainStore';
|
import { useMainStore } from '../../store/mainStore';
|
||||||
import { useApiStore } from '../../store/apiStore';
|
import { useApiStore } from '../../store/apiStore';
|
||||||
import { StationRoutesInfo, Train } from '../../typings/common';
|
import { StationRoutesInfo, TimetablePathElement, Train } from '../../typings/common';
|
||||||
|
import { TrainSchedulePoint } from './typings';
|
||||||
export interface TrainScheduleStop {
|
|
||||||
nameHtml: string;
|
|
||||||
nameRaw: string;
|
|
||||||
|
|
||||||
status: 'confirmed' | 'unconfirmed' | 'stopped';
|
|
||||||
type: string;
|
|
||||||
position: 'begin' | 'end' | 'en-route';
|
|
||||||
|
|
||||||
arrivalScheduled: number;
|
|
||||||
arrivalReal: number;
|
|
||||||
|
|
||||||
departureScheduled: number;
|
|
||||||
departureReal: number;
|
|
||||||
|
|
||||||
departureDelay: number;
|
|
||||||
arrivalDelay: number;
|
|
||||||
|
|
||||||
duration: number | null;
|
|
||||||
|
|
||||||
isActive: boolean;
|
|
||||||
isLastConfirmed: boolean;
|
|
||||||
isSBL: boolean;
|
|
||||||
|
|
||||||
sceneryName: string | null;
|
|
||||||
distance: number;
|
|
||||||
|
|
||||||
arrivalLine: string | null;
|
|
||||||
departureLine: string | null;
|
|
||||||
|
|
||||||
arrivalLineInfo?: StationRoutesInfo;
|
|
||||||
departureLineInfo?: StationRoutesInfo;
|
|
||||||
|
|
||||||
isExternal: boolean;
|
|
||||||
|
|
||||||
comments: string | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: { StopLabel, StockList },
|
components: { StopLabel, StockList },
|
||||||
@@ -163,33 +196,86 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
computed: {
|
methods: {
|
||||||
scheduleStops(): TrainScheduleStop[] {
|
getPathSceneryData(pathEl: TimetablePathElement) {
|
||||||
if (!this.train.timetableData) return [];
|
const sceneryData =
|
||||||
|
this.store.stationList?.find((sc) => sc.name == pathEl.stationName) ?? null;
|
||||||
|
|
||||||
const { timetablePath } = this.train.timetableData;
|
const activeScenery = this.apiStore.activeData?.activeSceneries?.find(
|
||||||
let currentPathIndex = 0;
|
(sc) => sc.stationName == pathEl.stationName
|
||||||
|
|
||||||
return (
|
|
||||||
this.train.timetableData?.followingStops.map((stop, i, arr) => {
|
|
||||||
const isExternal =
|
|
||||||
i < arr.length - 1 &&
|
|
||||||
stop.departureLine === timetablePath[currentPathIndex].departureRouteExt;
|
|
||||||
|
|
||||||
const sceneryName = timetablePath[currentPathIndex].stationName;
|
|
||||||
const sceneryInfo = this.apiStore.sceneryData.find((st) => st.name == sceneryName);
|
|
||||||
|
|
||||||
const arrivalLineInfo = sceneryInfo?.routesInfo.find(
|
|
||||||
(r) => r.routeName == stop.arrivalLine
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const departureLineInfo = sceneryInfo?.routesInfo.find(
|
const arrivalLineData = sceneryData?.generalInfo
|
||||||
(r) => r.routeName == stop.departureLine
|
? pathEl.arrivalRouteExt
|
||||||
);
|
? (sceneryData.generalInfo.routes.all.find(
|
||||||
|
(rt) => rt.routeName == pathEl.arrivalRouteExt
|
||||||
|
) ?? null)
|
||||||
|
: null
|
||||||
|
: null;
|
||||||
|
|
||||||
if (isExternal) currentPathIndex++;
|
const departureLineData = sceneryData?.generalInfo
|
||||||
|
? pathEl.departureRouteExt
|
||||||
|
? (sceneryData.generalInfo.routes.all.find(
|
||||||
|
(rt) => rt.routeName == pathEl.departureRouteExt
|
||||||
|
) ?? null)
|
||||||
|
: null
|
||||||
|
: null;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
generalInfo: sceneryData?.generalInfo ?? null,
|
||||||
|
isOnline:
|
||||||
|
activeScenery &&
|
||||||
|
(activeScenery.isOnline == 1 || activeScenery.lastSeen >= Date.now() - 60000),
|
||||||
|
arrivalLineData,
|
||||||
|
departureLineData
|
||||||
|
};
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
computed: {
|
||||||
|
scheduleStops() {
|
||||||
|
if (!this.train.timetableData) return [];
|
||||||
|
|
||||||
|
const { timetablePath, followingStops } = this.train.timetableData;
|
||||||
|
|
||||||
|
const stopRows: TrainSchedulePoint[] = [];
|
||||||
|
|
||||||
|
let currentPathIndex = 0;
|
||||||
|
let currentPath = timetablePath[0];
|
||||||
|
|
||||||
|
let pathData = this.getPathSceneryData(currentPath);
|
||||||
|
|
||||||
|
let arrivalLineInfo: StationRoutesInfo | null = null;
|
||||||
|
let departureLineInfo: StationRoutesInfo | null = null;
|
||||||
|
|
||||||
|
let isActive = false;
|
||||||
|
|
||||||
|
if (pathData?.departureLineData) {
|
||||||
|
arrivalLineInfo = pathData.departureLineData;
|
||||||
|
departureLineInfo = pathData.departureLineData;
|
||||||
|
}
|
||||||
|
|
||||||
|
followingStops.forEach((stop, i) => {
|
||||||
|
let isExternal = false;
|
||||||
|
|
||||||
|
if (stop.arrivalLine === currentPath.arrivalRouteExt) {
|
||||||
|
isExternal = true;
|
||||||
|
|
||||||
|
departureLineInfo = pathData?.arrivalLineData ?? null;
|
||||||
|
arrivalLineInfo = pathData.arrivalLineData;
|
||||||
|
}
|
||||||
|
|
||||||
|
const internalRouteInfo = stop.departureLine
|
||||||
|
? pathData?.generalInfo?.routes.all.find(
|
||||||
|
(route) => route.isInternal && route.routeName == stop.departureLine
|
||||||
|
)
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
if (internalRouteInfo) {
|
||||||
|
departureLineInfo = internalRouteInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
let rowData: TrainSchedulePoint = {
|
||||||
nameHtml: stop.stopName,
|
nameHtml: stop.stopName,
|
||||||
nameRaw: stop.stopNameRAW,
|
nameRaw: stop.stopNameRAW,
|
||||||
|
|
||||||
@@ -202,7 +288,9 @@ export default defineComponent({
|
|||||||
departureDelay: stop.departureDelay,
|
departureDelay: stop.departureDelay,
|
||||||
arrivalDelay: stop.arrivalDelay,
|
arrivalDelay: stop.arrivalDelay,
|
||||||
|
|
||||||
duration: stop.stopTime,
|
duration: stop.stopTime ?? 0,
|
||||||
|
type: stop.stopType,
|
||||||
|
distance: stop.stopDistance,
|
||||||
|
|
||||||
comments: stop.comments ?? null,
|
comments: stop.comments ?? null,
|
||||||
|
|
||||||
@@ -210,21 +298,67 @@ export default defineComponent({
|
|||||||
departureLine: stop.departureLine,
|
departureLine: stop.departureLine,
|
||||||
|
|
||||||
arrivalLineInfo: arrivalLineInfo,
|
arrivalLineInfo: arrivalLineInfo,
|
||||||
departureLineInfo: departureLineInfo,
|
departureLineInfo,
|
||||||
|
|
||||||
isExternal,
|
isExternal,
|
||||||
|
isActive,
|
||||||
|
|
||||||
type: stop.stopType,
|
|
||||||
distance: stop.stopDistance,
|
|
||||||
isActive: this.activeMinorStops.includes(i),
|
|
||||||
isLastConfirmed: this.lastConfirmed === i && !stop.terminatesHere,
|
|
||||||
isSBL: /sbl/gi.test(stop.stopName),
|
isSBL: /sbl/gi.test(stop.stopName),
|
||||||
position: stop.beginsHere ? 'begin' : stop.terminatesHere ? 'end' : 'en-route',
|
position: stop.beginsHere ? 'begin' : stop.terminatesHere ? 'end' : 'en-route',
|
||||||
sceneryName,
|
status: stop.confirmed ? 'confirmed' : stop.stopped ? 'stopped' : 'unconfirmed',
|
||||||
status: stop.confirmed ? 'confirmed' : stop.stopped ? 'stopped' : 'unconfirmed'
|
|
||||||
|
sceneryName: currentPath.stationName,
|
||||||
|
isSceneryOnline: pathData?.isOnline ?? false,
|
||||||
|
|
||||||
|
nextPointRef: null
|
||||||
};
|
};
|
||||||
}) ?? []
|
|
||||||
);
|
if (internalRouteInfo) {
|
||||||
|
arrivalLineInfo = departureLineInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
stopRows[stopRows.length - 1]?.status == 'confirmed' &&
|
||||||
|
rowData.status != 'confirmed' &&
|
||||||
|
rowData.status != 'stopped'
|
||||||
|
)
|
||||||
|
stopRows[stopRows.length - 1].isActive = true;
|
||||||
|
|
||||||
|
if (
|
||||||
|
stopRows[stopRows.length - 1]?.isActive == true &&
|
||||||
|
!/(^<strong>|, podg$)/.test(rowData.nameHtml)
|
||||||
|
)
|
||||||
|
rowData.isActive = true;
|
||||||
|
|
||||||
|
stopRows.push(rowData);
|
||||||
|
|
||||||
|
// Assign this row data object to the last one as reference
|
||||||
|
if (i != 0) {
|
||||||
|
stopRows[i - 1].nextPointRef = rowData;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (stop.departureLine === currentPath.departureRouteExt) {
|
||||||
|
// Reverse search for last scenery checkpoint
|
||||||
|
if (pathData?.departureLineData) {
|
||||||
|
stopRows[stopRows.length - 1].isExternal = true;
|
||||||
|
|
||||||
|
for (let i = stopRows.length - 1; i > 0; i--) {
|
||||||
|
stopRows[i].departureLineInfo = pathData.departureLineData;
|
||||||
|
|
||||||
|
if (/(^<strong>|, podg$)/.test(stopRows[i].nameHtml)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
stopRows[i].arrivalLineInfo = pathData.departureLineData;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
currentPath = timetablePath[++currentPathIndex];
|
||||||
|
pathData = this.getPathSceneryData(currentPath);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return stopRows;
|
||||||
},
|
},
|
||||||
|
|
||||||
lastConfirmed() {
|
lastConfirmed() {
|
||||||
@@ -256,18 +390,12 @@ export default defineComponent({
|
|||||||
|
|
||||||
return activeMinorStopList;
|
return activeMinorStopList;
|
||||||
}
|
}
|
||||||
},
|
|
||||||
|
|
||||||
methods: {
|
|
||||||
toggleShowState() {
|
|
||||||
this.$emit('click');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
$barClr: #b1b1b1;
|
$barClr: #b1b1b1;
|
||||||
$confirmedClr: #4ae24a;
|
$confirmedClr: #4ae24a;
|
||||||
@@ -285,10 +413,6 @@ $blinkAnim: 0.5s ease-in-out alternate infinite blink;
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.train-schedule {
|
|
||||||
padding: 1em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.schedule-wrapper {
|
.schedule-wrapper {
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
@@ -307,6 +431,10 @@ $blinkAnim: 0.5s ease-in-out alternate infinite blink;
|
|||||||
}
|
}
|
||||||
|
|
||||||
.stop {
|
.stop {
|
||||||
|
&[data-sbl='true'] {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
// Begin stop
|
// Begin stop
|
||||||
&[data-position='begin'] {
|
&[data-position='begin'] {
|
||||||
.node {
|
.node {
|
||||||
@@ -338,20 +466,19 @@ $blinkAnim: 0.5s ease-in-out alternate infinite blink;
|
|||||||
border-color: $haltClr;
|
border-color: $haltClr;
|
||||||
}
|
}
|
||||||
|
|
||||||
&[data-minor-stop-active='true'] {
|
// &[data-minor-stop-active='true'] {
|
||||||
.progress > .line {
|
// .progress > .line {
|
||||||
animation: $blinkAnim;
|
// animation: $blinkAnim;
|
||||||
}
|
// }
|
||||||
|
|
||||||
& + div {
|
// & + div {
|
||||||
.progress > .line_node-top {
|
// .progress > .line_node-top {
|
||||||
animation: $blinkAnim;
|
// animation: $blinkAnim;
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
}
|
// }
|
||||||
|
|
||||||
// Last confirmed outpost / checkpoint
|
&[data-is-active='true'] {
|
||||||
&[data-last-confirmed='true'] {
|
|
||||||
.progress > .line_connection {
|
.progress > .line_connection {
|
||||||
animation: $blinkAnim;
|
animation: $blinkAnim;
|
||||||
}
|
}
|
||||||
@@ -372,6 +499,7 @@ $blinkAnim: 0.5s ease-in-out alternate infinite blink;
|
|||||||
.progress > .node {
|
.progress > .node {
|
||||||
border-color: $confirmedClr;
|
border-color: $confirmedClr;
|
||||||
}
|
}
|
||||||
|
|
||||||
.progress > .line {
|
.progress > .line {
|
||||||
border-left: 2px solid $confirmedClr;
|
border-left: 2px solid $confirmedClr;
|
||||||
border-right: 2px solid $confirmedClr;
|
border-right: 2px solid $confirmedClr;
|
||||||
@@ -392,19 +520,22 @@ $blinkAnim: 0.5s ease-in-out alternate infinite blink;
|
|||||||
// Unused so far
|
// Unused so far
|
||||||
&[data-track-count-departure='2'] {
|
&[data-track-count-departure='2'] {
|
||||||
.progress > .line {
|
.progress > .line {
|
||||||
width: 6px;
|
width: 8px;
|
||||||
|
border-width: 3px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&[data-track-count-arrival='2'] {
|
&[data-track-count-arrival='2'] {
|
||||||
.progress > .line_node-top {
|
.progress > .line_node-top {
|
||||||
width: 6px;
|
width: 8px;
|
||||||
|
border-width: 3px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&[data-track-count-arrival='1'] {
|
&[data-track-count-arrival='1'] {
|
||||||
.progress > .line_node-top {
|
.progress > .line_node-top {
|
||||||
width: 4px;
|
width: 2px;
|
||||||
|
border-width: 2px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -532,8 +663,8 @@ $blinkAnim: 0.5s ease-in-out alternate infinite blink;
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
img {
|
img[data-tooltip] {
|
||||||
width: 1em;
|
cursor: help;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -13,10 +13,10 @@
|
|||||||
|
|
||||||
<transition name="dropdown-anim">
|
<transition name="dropdown-anim">
|
||||||
<div class="dropdown_wrapper" v-if="showOptions">
|
<div class="dropdown_wrapper" v-if="showOptions">
|
||||||
<h1 class="text--primary">
|
<h2 class="stats-title text--primary">
|
||||||
<img src="/images/icon-stats.svg" alt="Open filters icon" />
|
<img src="/images/icon-stats.svg" alt="Open filters icon" height="28" />
|
||||||
{{ $t('train-stats.title') }}
|
{{ $t('train-stats.title') }}
|
||||||
</h1>
|
</h2>
|
||||||
|
|
||||||
<hr style="margin: 0.5em 0" />
|
<hr style="margin: 0.5em 0" />
|
||||||
|
|
||||||
@@ -80,7 +80,11 @@
|
|||||||
<h3>{{ $t('train-stats.top-units') }}</h3>
|
<h3>{{ $t('train-stats.top-units') }}</h3>
|
||||||
|
|
||||||
<transition-group tag="ul" name="stats-anim">
|
<transition-group tag="ul" name="stats-anim">
|
||||||
<li class="badge stat-badge" v-for="top in stats.topUnits.slice(0, 7)" :key="top.name">
|
<li
|
||||||
|
class="badge stat-badge"
|
||||||
|
v-for="top in stats.topUnits.slice(0, 7)"
|
||||||
|
:key="top.name"
|
||||||
|
>
|
||||||
<span>{{ top.name }}</span>
|
<span>{{ top.name }}</span>
|
||||||
<span>{{ top.count }}</span>
|
<span>{{ top.count }}</span>
|
||||||
</li>
|
</li>
|
||||||
@@ -221,11 +225,11 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/dropdown.scss';
|
@use '../../styles/dropdown';
|
||||||
@import '../../styles/badge.scss';
|
@use '../../styles/badge';
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
h1 img {
|
.stats-title img {
|
||||||
vertical-align: text-bottom;
|
vertical-align: text-bottom;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -248,10 +252,13 @@ h3 {
|
|||||||
max-width: 600px;
|
max-width: 600px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@include smallScreen {
|
@include responsive.smallScreen{
|
||||||
h1,
|
|
||||||
.no-data {
|
.no-data {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.stats-title {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -1,5 +1,12 @@
|
|||||||
<template>
|
<template>
|
||||||
<transition name="status-anim" mode="out-in" tag="div" class="train-table">
|
<transition
|
||||||
|
name="status-anim"
|
||||||
|
mode="out-in"
|
||||||
|
tag="div"
|
||||||
|
class="train-table"
|
||||||
|
@scroll="onScroll"
|
||||||
|
ref="trainTableRef"
|
||||||
|
>
|
||||||
<div :key="apiStore.dataStatuses.connection">
|
<div :key="apiStore.dataStatuses.connection">
|
||||||
<div class="table-warning" key="offline" v-if="store.isOffline">
|
<div class="table-warning" key="offline" v-if="store.isOffline">
|
||||||
{{ $t('app.offline') }}
|
{{ $t('app.offline') }}
|
||||||
@@ -12,16 +19,8 @@
|
|||||||
>)
|
>)
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<transition-group name="list-anim" tag="ul">
|
<transition-group name="list-anim" tag="div" class="list_wrapper">
|
||||||
<li
|
<TrainTableItem v-for="train in trains" :key="train.id" :train="train" />
|
||||||
class="train-row"
|
|
||||||
v-for="train in trains"
|
|
||||||
:key="train.id"
|
|
||||||
>
|
|
||||||
<router-link class="a-block" :to="train.driverRouteLocation">
|
|
||||||
<TrainInfo :train="train" :extended="false" />
|
|
||||||
</router-link>
|
|
||||||
</li>
|
|
||||||
</transition-group>
|
</transition-group>
|
||||||
</div>
|
</div>
|
||||||
</transition>
|
</transition>
|
||||||
@@ -30,13 +29,15 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { defineComponent, inject, PropType, Ref } from 'vue';
|
import { defineComponent, inject, PropType, Ref } from 'vue';
|
||||||
import { useMainStore } from '../../store/mainStore';
|
import { useMainStore } from '../../store/mainStore';
|
||||||
import Loading from '../Global/Loading.vue';
|
|
||||||
import TrainInfo from './TrainInfo.vue';
|
|
||||||
import { Status, Train } from '../../typings/common';
|
|
||||||
import { useApiStore } from '../../store/apiStore';
|
import { useApiStore } from '../../store/apiStore';
|
||||||
|
import { Status, Train } from '../../typings/common';
|
||||||
|
|
||||||
|
import Loading from '../Global/Loading.vue';
|
||||||
|
import TrainTableItem from './TrainTableItem.vue';
|
||||||
|
import TrainInfo from './TrainInfo.vue';
|
||||||
|
|
||||||
export default defineComponent({
|
export default defineComponent({
|
||||||
components: { Loading, TrainInfo },
|
components: { Loading, TrainInfo, TrainTableItem },
|
||||||
|
|
||||||
props: {
|
props: {
|
||||||
trains: {
|
trains: {
|
||||||
@@ -45,6 +46,10 @@ export default defineComponent({
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
data: () => ({
|
||||||
|
scrollTop: 0
|
||||||
|
}),
|
||||||
|
|
||||||
setup() {
|
setup() {
|
||||||
const store = useMainStore();
|
const store = useMainStore();
|
||||||
const apiStore = useApiStore();
|
const apiStore = useApiStore();
|
||||||
@@ -64,6 +69,16 @@ export default defineComponent({
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
|
activated() {
|
||||||
|
(this.$refs['trainTableRef'] as HTMLElement).scrollTop = this.scrollTop;
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
onScroll(e: Event) {
|
||||||
|
this.scrollTop = (e.target as HTMLElement).scrollTop;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
computed: {
|
computed: {
|
||||||
dataStatus() {
|
dataStatus() {
|
||||||
if (this.store.isOffline) return Status.Data.Offline;
|
if (this.store.isOffline) return Status.Data.Offline;
|
||||||
@@ -78,8 +93,8 @@ export default defineComponent({
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import '../../styles/responsive.scss';
|
@use '../../styles/responsive';
|
||||||
@import '../../styles/animations.scss';
|
@use '../../styles/animations';
|
||||||
|
|
||||||
.train-table {
|
.train-table {
|
||||||
height: calc(100vh - 11em);
|
height: calc(100vh - 11em);
|
||||||
@@ -90,6 +105,10 @@ export default defineComponent({
|
|||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.list_wrapper {
|
||||||
|
padding: 2px; // ensures focused items outline visibility
|
||||||
|
}
|
||||||
|
|
||||||
.table-warning {
|
.table-warning {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
|
||||||
@@ -98,10 +117,4 @@ export default defineComponent({
|
|||||||
|
|
||||||
background: #1a1a1a;
|
background: #1a1a1a;
|
||||||
}
|
}
|
||||||
|
|
||||||
li.train-row {
|
|
||||||
background-color: var(--clr-secondary);
|
|
||||||
margin-bottom: 1em;
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -0,0 +1,75 @@
|
|||||||
|
<template>
|
||||||
|
<router-link :to="train.driverRouteLocation" class="train-item">
|
||||||
|
<div class="item-wrapper">
|
||||||
|
<TrainInfo :train="train" />
|
||||||
|
|
||||||
|
<div class="train-stats">
|
||||||
|
<StockList :trainStockList="train.stockList" :tractionOnly="true" />
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<span>{{ train.speed }}km/h</span>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<span> {{ train.length }}m</span>
|
||||||
|
•
|
||||||
|
<span> {{ (train.mass / 1000).toFixed(1) }}t</span>
|
||||||
|
<span v-if="train.stockList.length > 1">
|
||||||
|
•
|
||||||
|
{{ $t('trains.cars') }}: {{ train.stockList.length - 1 }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</router-link>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts">
|
||||||
|
import { PropType } from 'vue';
|
||||||
|
import { Train } from '../../typings/common';
|
||||||
|
import TrainInfo from './TrainInfo.vue';
|
||||||
|
import StockList from '../Global/StockList.vue';
|
||||||
|
|
||||||
|
defineProps({
|
||||||
|
train: {
|
||||||
|
type: Object as PropType<Train>,
|
||||||
|
required: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
@use '../../styles/responsive';
|
||||||
|
|
||||||
|
.train-item {
|
||||||
|
display: block;
|
||||||
|
background-color: #1a1a1a;
|
||||||
|
margin-bottom: 1em;
|
||||||
|
width: 100%;
|
||||||
|
padding: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-wrapper {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 2fr 1fr;
|
||||||
|
grid-template-rows: 1fr;
|
||||||
|
}
|
||||||
|
|
||||||
|
.train-stats {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
flex-direction: column;
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
|
line-height: 1.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
@include responsive.smallScreen {
|
||||||
|
.item-wrapper {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
gap: 1em 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { StationRoutesInfo } from "../../typings/common";
|
||||||
|
|
||||||
export enum TrainFilterSection {
|
export enum TrainFilterSection {
|
||||||
TRAIN_TYPE = 'TRAIN_TYPE',
|
TRAIN_TYPE = 'TRAIN_TYPE',
|
||||||
TIMETABLE_TYPE = 'TIMETABLE_TYPE',
|
TIMETABLE_TYPE = 'TIMETABLE_TYPE',
|
||||||
@@ -10,12 +12,14 @@ export const enum TrainFilterId {
|
|||||||
withComments = 'withComments',
|
withComments = 'withComments',
|
||||||
|
|
||||||
twr = 'twr',
|
twr = 'twr',
|
||||||
skr = 'skr',
|
tn = 'tn',
|
||||||
|
pn = 'pn',
|
||||||
common = 'common',
|
common = 'common',
|
||||||
|
|
||||||
passenger = 'passenger',
|
passenger = 'passenger',
|
||||||
freight = 'freight',
|
freight = 'freight',
|
||||||
other = 'other',
|
other = 'other',
|
||||||
|
|
||||||
noTimetable = 'noTimetable',
|
noTimetable = 'noTimetable',
|
||||||
withTimetable = 'withTimetable'
|
withTimetable = 'withTimetable'
|
||||||
}
|
}
|
||||||
@@ -38,7 +42,12 @@ export const trainFilters: TrainFilter[] = [
|
|||||||
isActive: true
|
isActive: true
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: TrainFilterId.skr,
|
id: TrainFilterId.tn,
|
||||||
|
section: TrainFilterSection.TRAIN_TYPE,
|
||||||
|
isActive: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: TrainFilterId.pn,
|
||||||
section: TrainFilterSection.TRAIN_TYPE,
|
section: TrainFilterSection.TRAIN_TYPE,
|
||||||
isActive: true
|
isActive: true
|
||||||
},
|
},
|
||||||
@@ -117,3 +126,76 @@ export const sorterOptions: TrainSorter[] = [
|
|||||||
value: 'długość'
|
value: 'długość'
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export interface TrainScheduleStop {
|
||||||
|
nameHtml: string;
|
||||||
|
nameRaw: string;
|
||||||
|
|
||||||
|
status: 'confirmed' | 'unconfirmed' | 'stopped';
|
||||||
|
type: string;
|
||||||
|
position: 'begin' | 'end' | 'en-route';
|
||||||
|
|
||||||
|
arrivalScheduled: number;
|
||||||
|
arrivalReal: number;
|
||||||
|
|
||||||
|
departureScheduled: number;
|
||||||
|
departureReal: number;
|
||||||
|
|
||||||
|
departureDelay: number;
|
||||||
|
arrivalDelay: number;
|
||||||
|
|
||||||
|
duration: number | null;
|
||||||
|
|
||||||
|
isActive: boolean;
|
||||||
|
isLastConfirmed: boolean;
|
||||||
|
isSBL: boolean;
|
||||||
|
|
||||||
|
sceneryName: string | null;
|
||||||
|
isSceneryOnline: boolean;
|
||||||
|
|
||||||
|
distance: number;
|
||||||
|
|
||||||
|
arrivalLine: string | null;
|
||||||
|
departureLine: string | null;
|
||||||
|
|
||||||
|
arrivalLineInfo?: StationRoutesInfo;
|
||||||
|
departureLineInfo?: StationRoutesInfo;
|
||||||
|
|
||||||
|
isExternal: boolean;
|
||||||
|
|
||||||
|
comments: string | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface TrainSchedulePoint {
|
||||||
|
nameHtml: string;
|
||||||
|
nameRaw: string;
|
||||||
|
|
||||||
|
status: 'confirmed' | 'unconfirmed' | 'stopped';
|
||||||
|
position: 'begin' | 'end' | 'en-route';
|
||||||
|
type: string;
|
||||||
|
duration: number;
|
||||||
|
distance: number;
|
||||||
|
arrivalScheduled: number;
|
||||||
|
arrivalReal: number;
|
||||||
|
departureScheduled: number;
|
||||||
|
departureReal: number;
|
||||||
|
|
||||||
|
comments: string | null;
|
||||||
|
|
||||||
|
arrivalDelay: number;
|
||||||
|
departureDelay: number;
|
||||||
|
arrivalLine: string | null;
|
||||||
|
departureLine: string | null;
|
||||||
|
|
||||||
|
arrivalLineInfo: StationRoutesInfo | null;
|
||||||
|
departureLineInfo: StationRoutesInfo | null;
|
||||||
|
|
||||||
|
isExternal: boolean,
|
||||||
|
|
||||||
|
isActive: boolean;
|
||||||
|
isSBL: boolean;
|
||||||
|
sceneryName: string | null;
|
||||||
|
isSceneryOnline: boolean;
|
||||||
|
|
||||||
|
nextPointRef: TrainSchedulePoint | null;
|
||||||
|
}
|
||||||
@@ -0,0 +1,95 @@
|
|||||||
|
{
|
||||||
|
"allCategories": [
|
||||||
|
"ROE", "ROJ", "ROS", "ROM",
|
||||||
|
"RPE", "RPJ", "RPS", "RPM",
|
||||||
|
"RME", "RMJ", "RMS", "RMM",
|
||||||
|
"RAE", "RAJ", "RAS", "RAM",
|
||||||
|
"MPE", "MPJ", "MPS", "MPM",
|
||||||
|
"MME", "MMJ", "MMS", "MMM",
|
||||||
|
"MOE", "MOJ", "MOS", "MOM",
|
||||||
|
"MHE", "MHJ", "MHS",
|
||||||
|
"EIE", "EIS",
|
||||||
|
"ENE", "ENS",
|
||||||
|
"ECE", "ECS",
|
||||||
|
"PWE", "PWM", "PWJ", "PWS",
|
||||||
|
"PXE", "PXM", "PXJ", "PXS",
|
||||||
|
"TCE", "TCS",
|
||||||
|
"TDE", "TDS",
|
||||||
|
"TGE", "TGS",
|
||||||
|
"TKE", "TKS",
|
||||||
|
"TME", "TMS",
|
||||||
|
"TNE", "TNS",
|
||||||
|
"TRE", "TRS",
|
||||||
|
"TSE", "TSS",
|
||||||
|
"THE", "THS",
|
||||||
|
"LPE", "LPS",
|
||||||
|
"LTE", "LTS",
|
||||||
|
"LSS",
|
||||||
|
"LZE", "LZS",
|
||||||
|
"ZNE", "ZNS",
|
||||||
|
"ZUE", "ZUS"
|
||||||
|
],
|
||||||
|
"regionNumbers": {
|
||||||
|
"Warszawa (1)": 1,
|
||||||
|
"Lublin (2)": 2,
|
||||||
|
"Kraków (3)": 3,
|
||||||
|
"Sosnowiec (4)": 4,
|
||||||
|
"Gdańsk (5)": 5,
|
||||||
|
"Wrocław (6)": 6,
|
||||||
|
"Poznań (7)": 7,
|
||||||
|
"Szczecin (8)": 8,
|
||||||
|
"Rezerwa (9)": 9
|
||||||
|
},
|
||||||
|
"sameRegions": {
|
||||||
|
"Losowy": [
|
||||||
|
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
|
||||||
|
],
|
||||||
|
"Warszawa (1)": [10, 11, 19, 91, 93, 97, 99],
|
||||||
|
"Lublin (2)": [20, 22, 29],
|
||||||
|
"Kraków (3)": [30, 33, 39],
|
||||||
|
"Sosnowiec (4)": [40, 44, 49, 94],
|
||||||
|
"Gdańsk (5)": [50, 55, 59, 90, 95, 96],
|
||||||
|
"Wrocław (6)": [66, 60, 69],
|
||||||
|
"Poznań (7)": [77, 70, 79],
|
||||||
|
"Szczecin (8)": [88, 80],
|
||||||
|
"Rezerwa (9)": [89, 92, 98]
|
||||||
|
},
|
||||||
|
"categoriesRules": {
|
||||||
|
"EI": [null, "00", "99"],
|
||||||
|
"EC": [null, "000", "049"],
|
||||||
|
"EN": [null, "000", "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"],
|
||||||
|
|
||||||
|
"TM": ["4", "000", "899"],
|
||||||
|
"TN": ["3", "000", "899"],
|
||||||
|
"TK": ["3", "000", "899"],
|
||||||
|
"TD": ["2", "000", "899"],
|
||||||
|
"TG": ["1", "000", "899"],
|
||||||
|
"TR": ["1", "000", "899"],
|
||||||
|
"TC": ["0", "000", "899"],
|
||||||
|
"TS": ["5", "000", "899"],
|
||||||
|
"TH": ["5", "000", "899"],
|
||||||
|
|
||||||
|
"LT": ["5", "000", "899"],
|
||||||
|
"LP": ["6", "000", "899"],
|
||||||
|
"LS": ["9", "000", "899"],
|
||||||
|
"LZ": ["9", "000", "899"],
|
||||||
|
|
||||||
|
"ZN": ["9", "000", "899"],
|
||||||
|
"ZU": ["9", "000", "899"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||