Compare commits

..

324 Commits

Author SHA1 Message Date
Spythere 16b3bb3683 Merge pull request #126 from Spythere/development
v1.29.2
2025-04-15 01:15:42 +02:00
Spythere 93e242c0f5 chore: additional styles 2025-04-15 01:10:31 +02:00
Spythere 861206a5ab chore: updated look of station stats; minor layout improvements 2025-04-14 21:37:59 +02:00
Spythere a47399fe1b chore: added driver stats percentage 2025-04-14 21:02:46 +02:00
Spythere 8e196c8279 chore: switched drivers filter from text to select input (TrainsView); updated inputs clear buttons 2025-04-14 19:24:28 +02:00
Spythere be55bac9fe chore: updated networking settings 2025-04-14 19:24:24 +02:00
Spythere c5e53057eb hotfix: post-upgrade adjustments 2025-03-29 16:08:34 +01:00
Spythere 4ba5d544af bump: v1.29.2 2025-03-26 18:04:19 +01:00
Spythere 22b6177560 restruct: updated sass version and rules 2025-03-26 18:04:00 +01:00
Spythere 9b6c6ee756 Merge pull request #125 from Spythere/development
v1.29.1
2025-03-24 14:58:12 +01:00
Spythere 829059d35b chore: added saving the routes visibility state in localStorage 2025-03-23 16:27:56 +01:00
Spythere b56e114ef9 restruct: scenery info components 2025-03-23 16:20:38 +01:00
Spythere 71b4cc3bdb fix: sticky table header bug 2025-03-22 16:10:37 +01:00
Spythere 8cc773ffb5 fix: filter card responsiveness 2025-03-22 16:08:35 +01:00
Spythere 427b4c03e4 chore: added hiding & showing internal routes in scenery view 2025-03-22 15:57:48 +01:00
Spythere 46dc43d652 bump: v1.29.1 2025-03-17 14:15:09 +01:00
Spythere 6435d12090 fix: resetting slider filters values 2025-03-17 14:14:13 +01:00
Spythere e41b8cfa98 chore: added internal station routes filters 2025-03-17 14:04:43 +01:00
Spythere bc81bb2a38 Merge pull request #124 from Spythere/development
v1.29.0 hotfixes
2025-02-13 18:39:53 +01:00
Spythere e6c064d15d fix: reworked train stop statuses descriptions and their tooltip styles 2025-02-13 18:38:32 +01:00
Spythere 4d1df5165c hotfix: proper schedule line tracks changing 2025-02-13 17:55:19 +01:00
Spythere 43ac2be3e7 Merge pull request #123 from Spythere/development
Development
2025-02-05 14:32:24 +01:00
Spythere 75c4e56183 fix: badge layout 2025-02-05 14:31:06 +01:00
Spythere 931f6b9fbd fix: train speed limits 2025-02-05 14:29:18 +01:00
Spythere 21fa1f8699 Merge pull request #122 from Spythere/development
hotfix: donation card actions layout
2025-02-04 23:34:13 +01:00
Spythere 877ef50a97 hotfix: donation card actions layout 2025-02-04 23:33:27 +01:00
Spythere 933be53630 Merge pull request #121 from Spythere/development
hotfix: lastSeen data filtering synchronization
2025-02-04 22:59:09 +01:00
Spythere eef4103960 hotfix: lastSeen data filtering synchronization 2025-02-04 22:57:27 +01:00
Spythere 9fafbe2c7f Merge pull request #120 from Spythere/development
v1.29.0
2025-02-04 20:50:29 +01:00
Spythere 666ba07307 chore: added SRJP external link 2025-02-04 18:14:58 +01:00
Spythere b63328f97c fix: journal double api fetching 2025-02-04 16:48:46 +01:00
Spythere 342127d541 chore: improved journal filtering by date 2025-02-04 16:42:48 +01:00
Spythere c6ab0d21de bump: v1.29.0 2025-02-04 15:15:21 +01:00
Spythere da4476bdf0 feat: saving train table scroll position 2025-02-04 15:14:58 +01:00
Spythere a950b4bef4 chore: calculating max train speed based on its mass 2025-02-04 15:11:11 +01:00
Spythere 5aa9297ec5 chore: resettings all train filter options on click 2025-02-04 14:17:04 +01:00
Spythere 0af49befc6 chore: changed layout of journal timetable badges 2025-02-04 14:12:47 +01:00
Spythere 4da0ab475b chore: changed placement of the offline icon 2025-02-04 14:12:20 +01:00
Spythere 1fa5934784 chore: added timetable vmax to journal 2025-02-04 00:05:42 +01:00
Spythere 5fb1a87b41 chore: added max timetable speed; route pairing fix 2025-02-03 23:51:47 +01:00
Spythere 8a5687cc01 chore: added different border width for double track routes 2025-02-03 22:48:38 +01:00
Spythere c5fe929b9a bump: v1.28.8 2025-02-02 22:22:23 +01:00
Spythere 5787deeaf8 restruct: added interal lines info to train schedule; minor fixes 2025-02-02 22:22:04 +01:00
Spythere 130732921b Merge pull request #119 from Spythere/development
v1.28.7
2025-01-28 14:51:24 +01:00
Spythere 1b2cd34e86 fix: responsive text center 2025-01-28 14:32:48 +01:00
Spythere 17bda9e6e7 chore: added scenery offline icon in active train schedule; icon improvements 2025-01-28 14:23:57 +01:00
Spythere c66ff8feed bump: v1.28.7 2025-01-15 16:26:02 +01:00
Spythere 027cdee25a fix: twr polish locale 2025-01-15 16:24:56 +01:00
Spythere 435cfb3b3f chore: added offline players indicators in the scenery view 2025-01-15 16:23:26 +01:00
Spythere 425241c8e7 Merge pull request #118 from Spythere/development
v1.28.6
2025-01-11 13:22:13 +01:00
Spythere f24f961d52 fix: warning notes display for TN & PN in journal 2025-01-11 13:09:45 +01:00
Spythere 4718eeeaaf bump: v1.28.6 2025-01-11 00:21:11 +01:00
Spythere 931fd7b21b feat: copying a railway stock of active drivers and timetable journal 2025-01-11 00:20:58 +01:00
Spythere bb79c5033a chore: added icon packs 2025-01-10 23:20:24 +01:00
Spythere ee290788dc Merge pull request #117 from Spythere/development
v1.28.5
2024-12-23 16:50:34 +01:00
Spythere a87d1060d3 chore: adjusted christmas dates 2024-12-23 16:45:46 +01:00
Spythere 1804d6d0f0 bump: v1.28.5 2024-12-23 16:44:59 +01:00
Spythere 77250e30c7 fix: preview tooltip fallback image 2024-12-23 16:44:42 +01:00
Spythere c5aefd03b8 Merge pull request #116 from Spythere/development
1.28.4 - minor fixes & updates
2024-12-20 16:27:08 +01:00
Spythere 2ec4694bd3 restruct: train info & timetable code 2024-12-20 15:51:16 +01:00
Spythere 729f66bcdb chore: added changing logo to christmas version 2024-12-20 15:46:34 +01:00
Spythere b746843086 Merge pull request #115 from Spythere/development
v1.28.4
2024-11-15 19:00:00 +01:00
Spythere cbbd06fecd bump: v1.28.4 2024-11-15 18:52:21 +01:00
Spythere 11e99b6af0 hotfix: stations sorting by 'no limit' 2024-11-15 18:51:34 +01:00
Spythere 31f4a2e5b2 Merge pull request #114 from Spythere/development
v1.28.3
2024-10-22 21:30:53 +02:00
Spythere 22514c3263 bump: v.1.28.3 2024-10-22 21:22:37 +02:00
Spythere 0df673467c fix: image loading styles 2024-10-22 21:22:22 +02:00
Spythere 6377e13809 refactor: footer component 2024-10-22 21:13:23 +02:00
Spythere 13fa633db4 chore: updated api image source 2024-10-22 20:58:20 +02:00
Spythere dd9661551c fix: typo 2024-10-22 20:58:08 +02:00
Spythere 495012a5ca Merge pull request #113 from Spythere/development
hotfix: data fetching
2024-10-02 14:59:34 +02:00
Spythere 3cfccb1bb4 hotfix: data fetching 2024-10-02 14:58:58 +02:00
Spythere d2a8cdb2b0 Merge pull request #112 from Spythere/development
v1.28.2
2024-10-02 14:31:39 +02:00
Spythere c33b5ef8c1 refactor: journal dispatcher filters 2024-10-01 16:40:11 +02:00
Spythere 52d1771c21 chore: added tn/pn filters for trains & timetables 2024-10-01 15:53:59 +02:00
Spythere cac4345683 chore: added journal timetable comments 2024-10-01 15:28:22 +02:00
Spythere 6fd9e21213 bump: v1.28.2 2024-09-30 22:33:45 +02:00
Spythere 421add1ec1 chore: added TN/PN freight types 2024-09-30 22:32:29 +02:00
Spythere 4ac92198b7 chore: api mock configuration files 2024-09-30 14:23:20 +02:00
Spythere 5105229eef chore: api mock endpoints 2024-09-30 11:50:08 +02:00
Spythere 2ec02080c3 chore: added internal api mocking for tests 2024-09-29 13:15:46 +02:00
Spythere 95b2a696e1 Merge pull request #111 from Spythere/development
fix: vehicle thumbnail names
2024-09-19 16:00:02 +02:00
Spythere 091e94e396 fix: vehicle thumbnail names 2024-09-19 15:48:55 +02:00
Spythere 43c939bf01 Merge pull request #110 from Spythere/development
v1.28.1
2024-09-18 20:09:22 +02:00
Spythere 8285d5c579 fix: tooltips width 2024-09-18 20:04:53 +02:00
Spythere 547248b478 chore: updown arrow fix 2024-09-18 16:08:54 +02:00
Spythere e1b9b37ac8 bump: v1.28.1 2024-09-18 15:57:56 +02:00
Spythere c8ec28292b chore: removed obsolete console logs 2024-09-18 15:57:45 +02:00
Spythere 3daf800a89 chore: added SBL icon & route tooltips 2024-09-18 15:55:33 +02:00
Spythere 69d9be0bb3 chore: minor thumbnail loading changes 2024-09-18 15:26:14 +02:00
Spythere f53f3a18fe Merge pull request #109 from Spythere/development
v1.28.0
2024-09-09 14:26:32 +02:00
Spythere fac8fced3e chore: removed journal list status animations 2024-09-09 14:11:35 +02:00
Spythere a3d9e68c8a chore: tooltip placing 2024-09-08 16:32:11 +02:00
Spythere b09761de58 chore: TWR & SKR badges fixes 2024-09-07 21:07:41 +02:00
Spythere 8ac2c68660 bump: v1.28.0 2024-09-07 17:28:29 +02:00
Spythere 4177c6e5f4 chore: displaying warning notes in driver view & journal timetables 2024-09-07 17:28:05 +02:00
Spythere b8f135a454 chore: thumbnail loading optimization 2024-09-06 15:23:22 +02:00
Spythere f0863b2459 chore: added static data hourly refresh 2024-09-05 17:00:15 +02:00
Spythere 55b4732992 chore: cleanup 2024-09-05 15:34:11 +02:00
Spythere 7b3dcea89e fix: missing category translations 2024-09-03 15:25:10 +02:00
Spythere f4b0c39185 chore: entry details backwards comp. 2024-09-03 15:18:59 +02:00
Spythere 275d602f97 chore: hiding entry details on history change 2024-09-03 14:43:16 +02:00
Spythere c93514fdf0 refactor: journal timetable entries 2024-09-03 14:29:59 +02:00
Spythere 0861d92e4b hotfix: class name 2024-09-02 23:41:39 +02:00
Spythere bdfd73f4be chore: stops design 2024-09-02 22:56:00 +02:00
Spythere df86364c51 chore: journal timetable stop labels 2024-09-02 22:39:41 +02:00
Spythere 631bb20c61 hotfix: resolved merge conflicts 2024-09-01 16:09:16 +02:00
Spythere bed79ed2d0 chore: translations 2024-09-01 16:07:40 +02:00
Spythere 2a07471e12 chore: displaying other driver's trains in the driver view 2024-08-31 13:51:22 +02:00
Spythere cfe188d0dc bump: v1.27.1 2024-08-29 16:04:14 +02:00
Spythere d9865be83e fix: views viewport height 2024-08-29 16:03:26 +02:00
Spythere 9155fd9f8d chore: driver view return button 2024-08-29 15:51:53 +02:00
Spythere 4674bf886e fix: manifest theme color 2024-08-29 15:47:23 +02:00
Spythere 289fd310df chore: viewport & routing fixes 2024-08-29 15:42:44 +02:00
Spythere b04797052f chore: expanded containers width in views, adjusted dropdowns 2024-08-24 16:16:32 +02:00
Spythere 7079f20791 Merge pull request #107 from Spythere/development
hotfix: journal stats badge styles
2024-08-24 00:25:35 +02:00
Spythere c244275aee hotfix: journal stats badge styles 2024-08-24 00:24:55 +02:00
Spythere fbc9785341 Merge pull request #105 from Spythere/development
v1.27.0
2024-08-24 00:20:08 +02:00
Spythere 9fd02c2336 chore: scenery view border radius 2024-08-23 16:24:21 +02:00
Spythere c031dd55c1 chore: removed return button 2024-08-23 16:22:54 +02:00
Spythere b0870699a4 hotfix: anchor style 2024-08-22 23:20:09 +02:00
Spythere a8da634b0e hotfix: TrainsView watcher causing routing problems 2024-08-22 23:19:42 +02:00
Spythere 8920b1e5e8 hotfix: view styles 2024-08-22 17:00:40 +02:00
Spythere 4fa1c05831 fix: input attrs 2024-08-22 16:44:51 +02:00
Spythere ecef2d5ee4 chore: backwards compatibility with train modal for ext. links 2024-08-22 16:37:47 +02:00
Spythere 1749871d08 fix: filters double click 2024-08-22 16:37:12 +02:00
Spythere 5545616706 chore: styling hotfixes and improvements 2024-08-22 02:28:40 +02:00
Spythere b35bb03868 merge: 'dominik-korsa-links' into development 2024-08-22 02:27:53 +02:00
Spythere b9521918cb hotfix: translation fix 2024-08-21 21:38:21 +02:00
Spythere 7ad17fc2c5 chore: style improvements & finishing touches 2024-08-21 21:33:46 +02:00
Spythere a80144cb1c chore: missing translations 2024-08-21 18:30:17 +02:00
Spythere 1227cdb94a chore: driver view links 2024-08-21 18:12:35 +02:00
Spythere b33594fd6f fix: journal list styles 2024-08-21 17:17:52 +02:00
Spythere 9f8656e590 fix: progress indicator 2024-08-21 17:06:50 +02:00
dominik-korsa 81cd165fe7 Use <router-link> instead of <tr> with click handler in StationTable 2024-08-21 14:42:03 +02:00
dominik-korsa 41e4b45599 Use <router-link> for driver journal button in TrainInfo 2024-08-21 13:54:48 +02:00
dominik-korsa 462dd7dd7a Replace all remaining uses of driverViewMixin with <router-link> 2024-08-21 13:49:31 +02:00
dominik-korsa 9837ae97e1 Use <router-link> instead of a @click handler in SceneryTimetable 2024-08-21 13:29:53 +02:00
dominik-korsa a818cd980b Use <router-link> instead of a @click handler in TrainTable 2024-08-21 12:58:47 +02:00
Spythere 573ebc233b chore: removed registering train modal 2024-08-21 02:06:38 +02:00
Spythere aae47c6abd fix: journal stats badges 2024-08-21 02:06:23 +02:00
Spythere 24c9b62162 feat: driver train view 2024-08-21 02:02:35 +02:00
Spythere 481d43b6d8 chore: selecting station checkpoint from url 2024-08-20 14:31:52 +02:00
Spythere 4969a433cc fix: train modal responsiveness & icons 2024-08-20 13:33:24 +02:00
Spythere 8a2b453dc6 chore: journal daily stats styling 2024-08-20 13:16:37 +02:00
Spythere 86d178ef56 chore: restored Pragotron link 2024-08-20 13:16:02 +02:00
Spythere 7769477508 chore: station stats styling 2024-08-20 00:14:44 +02:00
Spythere 551b60c733 fix: omitting "po" stops in timetable progress bar 2024-08-19 23:56:15 +02:00
Spythere 80a5b56785 feat: router links embeded into timetable stop names 2024-08-18 23:45:42 +02:00
Spythere 6bd62f13a1 chore: stats button responsiveness 2024-08-18 23:14:18 +02:00
Spythere 42591f6e76 chore: journal timetables styling improvements 2024-08-18 23:03:00 +02:00
Spythere 4ca0f09e75 chore: added ZG category 2024-08-18 22:58:22 +02:00
Spythere 02c3629c00 bump: v1.27.0 2024-08-18 01:43:51 +02:00
Spythere 9c4c806f0e chore: moved station stats to a dropdown 2024-08-18 01:43:27 +02:00
Spythere 58d6a97762 fix: twr/skr badges sizing 2024-08-17 15:51:36 +02:00
Spythere cd71c78eb4 chore: apicache typings 2024-08-17 15:51:25 +02:00
Spythere 300e70dcfe chore: journal timetables buttons alignment 2024-08-15 15:01:36 +02:00
Spythere 09b31f7914 Merge pull request #104 from Spythere/development
hotfix: warning icon placement
2024-08-10 23:41:34 +02:00
Spythere c29c3c6abe hotfix: warning icon placement 2024-08-10 23:41:00 +02:00
Spythere 916f6070ac Merge pull request #103 from Spythere/development
chore: added back merge gh workflow
2024-08-10 14:18:45 +02:00
Spythere a74ab6eb2a chore: added back merge gh workflow 2024-08-10 14:18:00 +02:00
Spythere 985c699ced Merge pull request #102 from Spythere/development
v1.26.1
2024-08-10 14:16:06 +02:00
Spythere 7e0e9146a5 fix: vehicle thumbnail cargo info 2024-08-10 14:08:32 +02:00
Spythere 30a0f05922 feat: journal dispatchers filtering by sc. hash 2024-08-10 14:00:25 +02:00
Spythere a30e04ca96 bump: v1.26.1 2024-08-09 15:24:48 +02:00
Spythere 1852d3e234 feat: category codes explanation tooltips 2024-08-09 15:24:26 +02:00
Spythere a17bf6c03f Merge branch 'development' 2024-08-09 14:12:07 +02:00
Spythere 766b08bc15 Merge branch 'master' of github.com:Spythere/stacjownik 2024-08-09 14:11:34 +02:00
Spythere cd1a4fa734 hotfix: checkpoint trains filtering 2024-08-06 14:18:26 +02:00
Spythere 619ce97b52 Merge pull request #101 from Spythere/development
v1.26.0
2024-08-05 16:11:07 +02:00
Spythere acbe761068 Merge branch 'development' 2024-08-05 16:02:20 +02:00
Spythere 47d35f335f chore: deleted merge workflow 2024-08-05 16:01:54 +02:00
Spythere 8fda8fa0df chore: package scripts 2024-08-05 15:59:54 +02:00
Spythere 71d697eda5 chore: changed workflow npm to yarn 2024-08-05 15:55:15 +02:00
Spythere f2b1fc5369 chore: disabled workflow on master push 2024-08-05 15:54:02 +02:00
Spythere 4a9b142e16 hotfix: lock files 2024-08-03 01:56:58 +02:00
Spythere 08d8bf3c57 bump: v1.26.0 2024-08-03 01:55:12 +02:00
Spythere 0ee90357aa chore: code structure 2024-08-03 01:53:36 +02:00
Spythere c8964dc20f chore: dispatcher history revamp & statuses 2024-08-02 02:00:44 +02:00
Spythere 6a62276d95 fix: vehicle preview loading 2024-08-01 19:26:25 +02:00
Spythere b8550eed9a chore: cleanup 2024-08-01 19:22:54 +02:00
Spythere 27b23ccc95 chore: lazy thumbnail loading & animations 2024-08-01 19:22:43 +02:00
Spythere b49517aded chore: packages upgrade 2024-07-24 18:55:41 +02:00
Spythere ed2b8be4dc chore: offline & fetching fixes 2024-07-24 17:52:20 +02:00
Spythere 54c1dbbf15 fix: stop labels statuses 2024-07-16 21:39:54 +02:00
Spythere 0ac7ba51e5 Merge pull request #100 from Spythere/development
v1.25.2
2024-07-12 16:13:06 +02:00
Spythere bdf85cd8ec bump: 1.25.2 2024-07-12 16:01:45 +02:00
Spythere 63b268d9b9 feat: added journal timetable path 2024-07-12 15:59:08 +02:00
Spythere d73c8ef112 fix: update modal won't open on first visit 2024-07-12 15:11:17 +02:00
Spythere 3d1c66b420 fix: cache control 2024-07-12 14:50:01 +02:00
Spythere b3f7108979 fix: detecting podg in timetables 2024-07-12 13:58:43 +02:00
Spythere feabfd29e0 Merge pull request #99 from Spythere/development
fix: recognizing timetables for sceneries with the same stop names
2024-07-09 20:33:16 +02:00
Spythere f17fedc976 fix: recognizing timetables for sceneries with the same stop names; optimization 2024-07-09 19:15:04 +02:00
Spythere c83c75e014 Merge pull request #98 from Spythere/development
hotfix: thumbnails v2 src
2024-07-08 22:12:41 +02:00
Spythere e57143f517 hotfix: thumbnails v2 src 2024-07-08 22:12:05 +02:00
Spythere fb45a783ee Merge pull request #97 from Spythere/development
v1.25.1
2024-07-08 21:40:50 +02:00
Spythere 71476e9552 bump: v1.25.1 2024-07-08 21:38:05 +02:00
Spythere 922a338143 hotfix: stock naming 2024-07-08 21:37:51 +02:00
Spythere 231d36e877 chore: adjusted for new vehicle thumbnails 2024-07-08 21:35:22 +02:00
Spythere 27d6ac9f14 Merge pull request #96 from Spythere/development
hotfix: scenery timetable train statuses
2024-06-11 20:56:25 +02:00
Spythere a6029da2cc hotfix: scenery timetable train statuses 2024-06-11 20:55:07 +02:00
Spythere a3f3790205 Merge pull request #95 from Spythere/development
hotfix: timetables for unknown sceneries
2024-06-10 20:19:20 +02:00
Spythere ebfb24f729 hotfix: timetables for unknown sceneries 2024-06-10 20:18:09 +02:00
Spythere e521736618 Merge pull request #94 from Spythere/development
hotfix: changed pwa strategy
2024-06-10 00:37:21 +02:00
Spythere fc7662e431 chore: changed pwa strategy 2024-06-10 00:36:30 +02:00
Spythere a459fdf178 Merge pull request #93 from Spythere/development
v1.25.0
2024-06-09 23:40:54 +02:00
Spythere 4e7fba89ee chore: improved stop label information 2024-06-09 00:58:45 +02:00
Spythere 6084e5876d chore: changed default history mode 2024-06-08 21:38:05 +02:00
Spythere 44f548c7b7 chore: scenery history locales 2024-06-08 21:37:28 +02:00
Spythere 59a5fbe5ac chore: adjusted to new version of API vehicles data 2024-06-08 20:53:22 +02:00
Spythere c252213ed9 hotfix 2024-06-07 18:31:20 +02:00
Spythere fb56378f18 chore: redesigned scenery history tables 2024-06-07 16:44:09 +02:00
Spythere e9635eae06 chore: redesigned train schedule list 2024-06-06 17:11:52 +02:00
Spythere 1fc98a8f99 chore: added test data mocks 2024-06-06 14:41:54 +02:00
Spythere c9de1a48ce chore: scenery timetables history translation; layout fixes 2024-06-06 14:19:17 +02:00
Spythere fee9774f88 chore: layout fixes 2024-06-06 14:12:21 +02:00
Spythere 7c974e8d0e bump: 1.25.0 2024-06-06 14:04:07 +02:00
Spythere c84fbbcf42 chore: added scenery timetables history modes 2024-06-05 20:03:05 +02:00
Spythere 45af649505 chore: changes in scenery view layout 2024-06-05 16:01:17 +02:00
Spythere 6c1e00d002 chore: layout & design fixes 2024-06-04 15:57:17 +02:00
Spythere 69ff85cfb1 chore: added route electrification indicators in train schedule 2024-06-03 22:26:58 +02:00
Spythere bdc2ca784c chore: missing translations 2024-06-03 21:37:33 +02:00
Spythere dbd73d448d chore: added active train's rolling stock vmax 2024-06-03 20:09:15 +02:00
Spythere 26b1ec246d chore: added extra data to vehicles tooltip 2024-06-03 18:10:45 +02:00
Spythere 8190dfa2cb chore: fetching & caching vehicles data information 2024-06-03 01:31:31 +02:00
Spythere 44df685606 Merge pull request #92 from Spythere/development
v1.24.4
2024-05-30 14:38:04 +02:00
Spythere 785a42b849 hotfix: detecting user timetable status at checkpoints 2024-05-30 14:29:09 +02:00
Spythere ccfcca8728 hotfix: scenery timetable duplicating 2024-05-30 14:24:18 +02:00
Spythere d9a7ba122c Merge pull request #91 from Spythere/development
v1.24.3
2024-05-26 01:44:45 +02:00
Spythere bf8d4a9ef4 chore: global font sizing; chore: train modal dvh 2024-05-25 18:06:01 +02:00
Spythere 6ea1e91d1d hotfix: card positioning 2024-05-25 17:57:25 +02:00
Spythere 813b557455 chore: improved card positioning 2024-05-25 17:55:18 +02:00
Spythere 834b14da69 fix: card dvh 2024-05-25 17:26:27 +02:00
Spythere c809b2146d chore: locale update 2024-05-25 17:12:19 +02:00
Spythere 33b98ca313 chore: added text color for active filters info 2024-05-25 17:11:28 +02:00
Spythere bcb9c63cb0 chore: reactive hiding body scroll on modal 2024-05-25 17:05:41 +02:00
Spythere 17d77a80d8 bump: 1.24.3 2024-05-25 16:02:40 +02:00
Spythere 65b159f8fd fix: scenery timetable duplicates; fix: not opening train modal for queries 2024-05-25 16:02:20 +02:00
Spythere 063d5283e4 Merge pull request #90 from Spythere/development
v1.24.2
2024-05-24 13:56:39 +02:00
Spythere 29de1b3c4b chore: scenery view layout 2024-05-24 13:52:42 +02:00
Spythere f0c02bf12e chore: pwa adjustments 2024-05-24 13:43:29 +02:00
Spythere 8aa23468b3 chore: changed station stats median to avg 2024-05-23 15:53:18 +02:00
Spythere 4c1fcf710b refactor: global modals to cards 2024-05-23 15:01:30 +02:00
Spythere a529d6e9eb chore: changed no stations message 2024-05-23 14:08:42 +02:00
Spythere 9fc602e08f chore: filters improvements 2024-05-22 15:41:33 +02:00
Spythere 56e40bd84b bump: version (1.24.2) 2024-05-21 16:17:41 +02:00
Spythere a5b5df7452 refactor: restructured station filters 2024-05-21 16:17:23 +02:00
Spythere 1a8da02ced chore: checkpoints detection fix 2024-05-19 23:42:06 +02:00
Spythere 7e75fa2516 chore: checkpoints hotfix 2024-05-19 23:12:07 +02:00
Spythere 3ed2c09184 chore: checkpoints filtering 2024-05-19 23:05:57 +02:00
Spythere 6901c3d2b4 chore: hotfix 2024-05-19 22:30:21 +02:00
Spythere 8417754403 refactor: optimization of train schedules 2024-05-19 19:50:01 +02:00
Spythere de5c57181a Merge pull request #89 from Spythere/development
v1.24.1
2024-05-16 23:43:39 +02:00
Spythere d91d4cc6a8 fix: station stats spawn count regions 2024-05-16 23:42:35 +02:00
Spythere 9a5fd4d670 chore: version bump 2024-05-16 23:29:56 +02:00
Spythere 4202a55673 chore: updated pwa strategies 2024-05-16 21:36:16 +02:00
Spythere 5181e8f4af chore: fix journal refresh date visibility 2024-05-16 20:06:02 +02:00
Spythere e117f62fcb chore: added station filters (scenery types); pwa adjustments 2024-05-16 19:59:43 +02:00
Spythere e0036bf969 chore: filters & stats fixes 2024-05-15 18:40:42 +02:00
Spythere 1f457d6389 Merge pull request #88 from Spythere/development
hotfix: minor adjustments for new simulator version (2024.1.1)
2024-05-13 15:05:28 +02:00
Spythere eb5b94c9f6 chore: vehicle images hotfixes 2024-05-13 15:02:15 +02:00
Spythere 328e8c0573 chore: fixed stock fallback thumbnnail 2024-05-13 14:54:21 +02:00
Spythere 9f58ae5428 Merge pull request #87 from Spythere/development
hotfix: modal positioning
2024-05-12 15:23:30 +02:00
Spythere 44f6cf4232 Merge branch 'development' 2024-05-12 15:22:28 +02:00
Spythere ebd0eeb8c4 hotfix: modal positioning 2024-05-12 15:22:03 +02:00
Spythere fa656c2f26 Merge pull request #86 from Spythere/development
v1.24.0
2024-05-12 15:14:22 +02:00
Spythere 0cc3a12d1d fix: modal responsiveness 2024-05-12 14:55:35 +02:00
Spythere 392a6437f8 feature: current users tooltip 2024-05-09 17:19:22 +02:00
Spythere 122532f0ed chore: general fixes 2024-05-09 16:40:53 +02:00
Spythere 366ff91f60 hotfix: update modal 2024-05-08 20:12:07 +02:00
Spythere a0496736dd chore: modals update 2024-05-08 20:04:41 +02:00
Spythere f974120e87 fix: lock files 2024-05-08 18:42:33 +02:00
Spythere abd8b8178b chore: vue deep selector 2024-05-08 16:42:04 +02:00
Spythere f1fcde8459 feat: update modal 2024-05-08 16:41:14 +02:00
Spythere b3289d6aab chore: region dropdown fixes 2024-05-08 15:16:20 +02:00
Spythere 6481a4a3b0 chore: design improvements 2024-05-08 15:10:40 +02:00
Spythere 05dc268526 fix: spawns detection 2024-05-06 18:18:15 +02:00
Spythere 669acc98d2 chore: station stats translation 2024-05-06 18:16:30 +02:00
Spythere 3371b661c2 fix: ufactor calc 2024-05-06 17:53:07 +02:00
Spythere 871b2c0221 feature: open spawns tooltip 2024-05-06 17:36:23 +02:00
Spythere d366a877a4 refactor: popups -> tooltips 2024-05-06 16:37:56 +02:00
Spythere 405aab96bd feature: stations stats 2024-05-05 13:34:43 +02:00
Spythere f29c160000 fix: lock files 2024-05-04 14:47:30 +02:00
Spythere a2de0e2030 refactor: types & performance 2024-05-04 14:43:34 +02:00
Spythere 7dd1c06f3f chore: accessibility of filters 2024-05-03 19:29:10 +02:00
Spythere ff041b9aaf bump(version): 1.24.0 2024-05-03 19:02:49 +02:00
Spythere 4782dba444 feat(app): added min route speed & max route speed station filters 2024-05-03 19:02:16 +02:00
Spythere d6b8d032d6 fix(app): improved data fetching scheduler 2024-05-03 19:02:13 +02:00
Spythere c16616330c chore(packages): update & cleanup 2024-05-03 18:01:54 +02:00
Spythere 57cec8bfe7 chore: pwa adjustments 2024-05-03 17:49:54 +02:00
Spythere 6bea340e19 chore(pwa): changed sceneries cache to cachefirst 2024-05-01 19:37:51 +02:00
Spythere c181cf7e64 fix(workflows): release color 2024-04-27 01:11:38 +02:00
Spythere 8e4ae64cd3 chore(workflows): added release discord webhook notification 2024-04-15 15:13:22 +02:00
Spythere 5750490f01 refactor: journals 2024-04-08 23:21:50 +02:00
Spythere 3ef27e1d69 Merge pull request #85 from Spythere/development
Wersja 1.23.1
2024-04-01 13:00:28 +02:00
Spythere f53993c717 hotfix 2024-03-31 21:55:33 +02:00
Spythere 235c16e30f train modal 2024-03-31 21:37:14 +02:00
Spythere c3533f07ad literówka 2024-03-30 17:48:34 +01:00
Spythere d05579c5ee popupy 2024-03-30 13:24:39 +01:00
Spythere c8f53c2f06 hotfixy designu 2024-03-30 00:18:54 +01:00
Spythere b44f88ebcd src miniaturek 2024-03-29 23:37:26 +01:00
Spythere 7805d1350c responsywność 2024-03-29 23:35:56 +01:00
Spythere b17bd19433 zmiana położenia przycisku RJ ONLINE w dzienniku 2024-03-29 23:23:14 +01:00
Spythere c12a6cbacd zmiana rozłożenia elementów w modalu aktywnego pociągu 2024-03-29 23:21:15 +01:00
Spythere ba650238db poprawki rozmieszczenia popupu 2024-03-29 23:04:08 +01:00
Spythere d5ec9919e2 update modal (wip) 2024-03-29 20:34:56 +01:00
Spythere 20cd393e05 Merge pull request #84 from Spythere/development
Wersja 1.23.0
2024-03-24 01:30:03 +01:00
Spythere 31e65c09d6 hotfix: podgląd pojazdów 2024-03-24 00:05:39 +01:00
Spythere fb2348e774 hotfixy designu 2024-03-23 23:55:18 +01:00
Spythere 1ec75bda70 poprawki do popupów 2024-03-23 16:47:57 +01:00
Spythere 6b6b837dde bump: v1.23.0 2024-03-23 00:01:15 +01:00
Spythere 66a02d76bd dodano odnośnik do dziennika RJ maszynisty 2024-03-23 00:00:52 +01:00
Spythere c7162dbd14 dodano dymki kontekstowe oraz podgląd pojazdu 2024-03-22 23:41:43 +01:00
Spythere 1cfe073bab Merge pull request #83 from Spythere/development
Wersja 1.22.3
2024-03-17 18:38:27 +01:00
Spythere e3b72c81ea bump: 1.22.3 2024-03-17 18:37:58 +01:00
Spythere 5552995564 fix: duplikujące się aktywne RJ scenerii 2024-03-17 18:37:45 +01:00
Spythere 623d5dd2ce fix: RJ scenerii offline 2024-03-17 17:33:19 +01:00
Spythere 6992b998a8 Merge pull request #82 from Spythere/development
Wersja 1.22.2
2024-03-17 16:47:52 +01:00
Spythere 669975c68e hotfixy 2024-03-17 16:42:35 +01:00
Spythere 084823de44 fix pobierania danych 2024-03-16 22:13:38 +01:00
Spythere f62d6982e5 Merge pull request #81 from Spythere/development
Wersja 1.22.1
2024-03-12 16:51:26 +01:00
Spythere 1c9b54b578 bump: 1.22.1 2024-03-11 23:50:00 +01:00
Spythere 0f4e5ee5f3 fix: niepoprawne miniaturki pojazdów 2024-03-11 23:49:45 +01:00
Spythere 29b5e715fa hotfixy wyglądu 2024-03-11 23:43:10 +01:00
176 changed files with 27994 additions and 29877 deletions
+2 -5
View File
@@ -1,6 +1,3 @@
# This file was auto-generated by the Firebase CLI
# https://github.com/firebase/firebase-tools
name: Deploy to Firebase Hosting on merge name: Deploy to Firebase Hosting on merge
'on': 'on':
push: push:
@@ -11,10 +8,10 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- run: npm ci && npm run build - run: yarn && yarn build
- uses: FirebaseExtended/action-hosting-deploy@v0 - uses: FirebaseExtended/action-hosting-deploy@v0
with: with:
repoToken: '${{ secrets.GITHUB_TOKEN }}' repoToken: '${{ secrets.GITHUB_TOKEN }}'
firebaseServiceAccount: '${{ secrets.FIREBASE_SERVICE_ACCOUNT_STACJOWNIK_TD2 }}' firebaseServiceAccount: '${{ secrets.FIREBASE_SERVICE_ACCOUNT_STACJOWNIK_TD2 }}'
channelId: live channelId: live
projectId: stacjownik-td2 projectId: stacjownik-td2
@@ -9,7 +9,7 @@ jobs:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
- run: npm ci && npm run build - run: yarn && yarn build
- uses: FirebaseExtended/action-hosting-deploy@v0 - uses: FirebaseExtended/action-hosting-deploy@v0
with: with:
repoToken: '${{ secrets.GITHUB_TOKEN }}' repoToken: '${{ secrets.GITHUB_TOKEN }}'
@@ -0,0 +1,17 @@
on:
release:
types: [published]
jobs:
github-releases-to-discord:
runs-on: ubuntu-latest
steps:
- name: Checkout
uses: actions/checkout@v3
- name: Github Releases To Discord
uses: SethCohen/github-releases-to-discord@v1.13.1
with:
webhook_url: ${{ secrets.WEBHOOK_URL }}
color: "15844367"
footer_title: "Changelog - Stacjownik"
footer_timestamp: true
+5 -1
View File
@@ -33,6 +33,10 @@ node_modules
# Env # Env
.env .env
.env.*
.fake .fake
.ionide .ionide
# api-mock
/api-mock/endpoints/
+33
View File
@@ -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();
+28
View File
@@ -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...');
});
+18
View File
@@ -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"
}
}
+481
View File
@@ -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==
-1
View File
@@ -19,7 +19,6 @@
<link rel="manifest" href="/site.webmanifest" /> <link rel="manifest" href="/site.webmanifest" />
<link rel="mask-icon" href="/safari-pinned-tab.svg" color="#5bbad5" /> <link rel="mask-icon" href="/safari-pinned-tab.svg" color="#5bbad5" />
<meta name="msapplication-TileColor" content="#da532c" /> <meta name="msapplication-TileColor" content="#da532c" />
<meta name="theme-color" content="#222222" />
<link rel="icon" href="favicon.ico" /> <link rel="icon" href="favicon.ico" />
+2546 -10534
View File
File diff suppressed because it is too large Load Diff
+20 -22
View File
@@ -1,11 +1,15 @@
{ {
"name": "stacjownik", "name": "stacjownik",
"version": "1.22.0", "version": "1.29.2",
"private": true, "private": true,
"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": "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",
"preview": "yarn build && vite preview", "preview": "yarn build && vite preview",
"type-check": "vue-tsc --noEmit -p tsconfig.app.json --composite false", "type-check": "vue-tsc --noEmit -p tsconfig.app.json --composite false",
"lint": "eslint . --ext .vue,.js,.jsx,.cjs,.mjs,.ts,.tsx,.cts,.mts --fix --ignore-path .gitignore", "lint": "eslint . --ext .vue,.js,.jsx,.cjs,.mjs,.ts,.tsx,.cts,.mts --fix --ignore-path .gitignore",
@@ -14,31 +18,25 @@
"dependencies": { "dependencies": {
"core-js": "^3.32.2", "core-js": "^3.32.2",
"dotenv": "^16.3.1", "dotenv": "^16.3.1",
"firebase": "^10.4.0",
"howler": "^2.2.4",
"pinia": "^2.1.6", "pinia": "^2.1.6",
"sass": "^1.67.0", "sass": "^1.67.0",
"socket.io-client": "^4.7.4", "showdown": "^2.1.0",
"vue": "^3.3.4", "vue": "^3.3.4",
"vue-i18n": "^9.4.1", "vue-i18n": "^9.4.1",
"vue-router": "^4.2.4" "vue-router": "^4.4.0"
}, },
"devDependencies": { "devDependencies": {
"@rushstack/eslint-patch": "^1.3.3", "@types/node": "^22.13.13",
"@types/node": "^20.6.2", "@types/showdown": "^2.0.6",
"@vite-pwa/assets-generator": "^0.0.10", "@vite-pwa/assets-generator": "^0.2.4",
"@vitejs/plugin-vue": "^4.3.4", "@vitejs/plugin-vue": "^5.1.0",
"@vue/eslint-config-prettier": "^8.0.0", "@vue/tsconfig": "^0.5.1",
"@vue/eslint-config-typescript": "^12.0.0", "axios": "^1.7.2",
"@vue/tsconfig": "^0.4.0", "prettier": "^3.3.3",
"axios": "^1.5.0", "typescript": "^5.5.4",
"eslint": "^8.49.0", "vite": "^5.3.4",
"eslint-plugin-vue": "^9.17.0", "vite-plugin-pwa": "^0.20.0",
"prettier": "^3.0.3", "vue-tsc": "^2.0.28"
"typescript": "^5.2.2",
"vite": "^4.4.9",
"vite-plugin-pwa": "^0.16.5",
"vue-tsc": "^1.8.11"
}, },
"browserslist": [ "browserslist": [
"> 1%", "> 1%",
File diff suppressed because it is too large Load Diff
+6243
View File
File diff suppressed because it is too large Load Diff
+19
View File
@@ -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; }
+19
View File
@@ -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; }
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
+3
View File
@@ -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

+5
View File
@@ -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

+5
View File
@@ -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

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 30 KiB

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 16 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 20 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 23 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 23 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 23 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 21 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 26 KiB

+1 -1
View File
@@ -13,7 +13,7 @@
"type": "image/png" "type": "image/png"
} }
], ],
"theme_color": "#ffc014", "theme_color": "#4d4d4d",
"background_color": "#4d4d4d", "background_color": "#4d4d4d",
"display": "standalone", "display": "standalone",
"start_url": "." "start_url": "."
-71
View File
@@ -1,71 +0,0 @@
@import './styles/responsive.scss';
@import './styles/variables.scss';
@import './styles/global.scss';
@import './styles/animations.scss';
.route {
margin: 0 0.2em;
&-active,
&[data-active='true'] {
color: $accentCol;
font-weight: bold;
}
}
// APP
#app {
color: white;
font-size: 1rem;
overflow-x: hidden;
@include smallScreen() {
font-size: calc(0.65rem + 0.8vw);
}
@include screenLandscape() {
font-size: calc(0.45rem + 0.8vw);
}
}
// CONTAINER
.app_container {
display: grid;
grid-template-rows: auto 1fr auto;
grid-template-columns: 100%;
min-height: 100vh;
}
.app_main {
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
footer.app_footer {
max-width: 100%;
padding: 0.5em;
img {
width: 1.1em;
vertical-align: text-bottom;
}
z-index: 10;
background: #111;
color: white;
text-align: center;
vertical-align: middle;
}
+113 -64
View File
@@ -1,11 +1,12 @@
<template> <template>
<div class="app_container"> <div class="app_container">
<transition name="modal-anim"> <UpdateCard
<keep-alive> :is-update-card-open="isUpdateCardOpen"
<TrainModal v-if="store.chosenModalTrainId" /> @toggle-card="() => (isUpdateCardOpen = false)"
</keep-alive> />
</transition>
<Tooltip />
<AppHeader :current-lang="currentLang" @change-lang="changeLang" /> <AppHeader :current-lang="currentLang" @change-lang="changeLang" />
<main class="app_main"> <main class="app_main">
@@ -16,37 +17,33 @@
</router-view> </router-view>
</main> </main>
<footer class="app_footer"> <AppFooter
&copy; :version="VERSION"
<a href="https://td2.info.pl/profile/?u=20777" target="_blank">Spythere</a> :is-on-production-host="isOnProductionHost"
{{ new Date().getUTCFullYear() }} | :is-update-card-open="isUpdateCardOpen"
<a :href="releaseURL" target="_blank">v{{ VERSION }}{{ isOnProductionHost ? '' : 'dev' }}</a> @open-update-card="() => (isUpdateCardOpen = true)"
<br /> />
<a href="https://discord.gg/x2mpNN3svk">
<img src="/images/icon-discord.png" alt="" />&nbsp;<b>{{ $t('footer.discord') }}</b>
</a>
<div style="display: none">&int; ukryta taktyczna całka do programowania w HTMLu</div>
</footer>
</div> </div>
</template> </template>
<script lang="ts"> <script lang="ts">
import { defineComponent, watch } from 'vue'; import { defineComponent } from 'vue';
import axios from 'axios'; import axios from 'axios';
import { version } from '.././package.json';
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 Clock from './components/App/Clock.vue';
import { useMainStore } from './store/mainStore';
import StatusIndicator from './components/App/StatusIndicator.vue'; import StatusIndicator from './components/App/StatusIndicator.vue';
import AppHeader from './components/App/AppHeader.vue'; import AppHeader from './components/App/AppHeader.vue';
import TrainModal from './components/TrainsView/TrainModal.vue'; import Tooltip from './components/Tooltip/Tooltip.vue';
import UpdateCard from './components/App/UpdateCard.vue';
import StorageManager from './managers/storageManager'; import StorageManager from './managers/storageManager';
import { useApiStore } from './store/apiStore'; import AppFooter from './components/App/AppFooter.vue';
import { Status } from './typings/common';
const STORAGE_VERSION_KEY = 'app_version'; const STORAGE_VERSION_KEY = 'app_version';
@@ -55,16 +52,20 @@ export default defineComponent({
Clock, Clock,
StatusIndicator, StatusIndicator,
AppHeader, AppHeader,
TrainModal AppFooter,
UpdateCard,
Tooltip
}, },
data: () => ({ data: () => ({
VERSION: version, VERSION: version,
store: useMainStore(), store: useMainStore(),
apiStore: useApiStore(), apiStore: useApiStore(),
tooltipStore: useTooltipStore(),
isUpdateCardOpen: false,
currentLang: 'pl', currentLang: 'pl',
releaseURL: '',
isOnProductionHost: location.hostname == 'stacjownik-td2.web.app' isOnProductionHost: location.hostname == 'stacjownik-td2.web.app'
}), }),
@@ -73,41 +74,45 @@ export default defineComponent({
}, },
async mounted() { async mounted() {
watch( window.addEventListener('mousemove', (e: MouseEvent) => this.tooltipStore.handle(e));
() => this.store.blockScroll, window.addEventListener('mousedown', () => this.tooltipStore.hide());
(value) => {
if (value) document.body.classList.add('no-scroll');
else document.body.classList.remove('no-scroll');
}
);
}, },
methods: { methods: {
init() { init() {
if (!this.isOnProductionHost) document.title = 'Stacjownik Dev';
this.loadLang(); this.loadLang();
this.setReleaseURL();
this.setupOfflineHandling(); this.setupOfflineHandling();
this.checkAppVersion(); this.checkAppVersion();
this.apiStore.setupAPIData(); this.apiStore.setupAPIData();
if (!this.isOnProductionHost) document.title = 'Stacjownik Dev';
}, },
checkAppVersion() { async checkAppVersion() {
if (import.meta.env.DEV) {
this.store.isNewUpdate = true;
return;
}
const storageVersion = StorageManager.getStringValue(STORAGE_VERSION_KEY); const storageVersion = StorageManager.getStringValue(STORAGE_VERSION_KEY);
if (storageVersion === undefined || storageVersion != version) { try {
this.store.isNewUpdate = true; const releaseData = await (
await axios.get('https://api.github.com/repos/Spythere/stacjownik/releases/latest')
).data;
StorageManager.setStringValue(STORAGE_VERSION_KEY, version); 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() { setupOfflineHandling() {
@@ -128,6 +133,7 @@ export default defineComponent({
handleOnlineMode() { handleOnlineMode() {
this.store.isOffline = false; this.store.isOffline = false;
this.apiStore.dataStatuses.connection = Status.Data.Loading;
this.apiStore.connectToAPI(); this.apiStore.connectToAPI();
}, },
@@ -139,21 +145,6 @@ export default defineComponent({
StorageManager.setStringValue('lang', lang); StorageManager.setStringValue('lang', lang);
}, },
async setReleaseURL() {
try {
const releaseData = await (
await axios.get('https://api.github.com/repos/Spythere/stacjownik/releases/latest')
).data;
if (!releaseData) return;
this.releaseURL = releaseData.html_url;
} catch (error) {
console.error(`Wystąpił błąd podczas pobierania danych z API GitHuba: ${error}`);
return;
}
},
loadLang() { loadLang() {
const storageLang = StorageManager.getStringValue('lang'); const storageLang = StorageManager.getStringValue('lang');
@@ -166,7 +157,7 @@ export default defineComponent({
const naviLanguage = window.navigator.language.toString(); const naviLanguage = window.navigator.language.toString();
if (naviLanguage.includes('en')) { if (naviLanguage.startsWith('en')) {
this.changeLang('en'); this.changeLang('en');
return; return;
} }
@@ -175,4 +166,62 @@ export default defineComponent({
}); });
</script> </script>
<style lang="scss" src="./App.scss"></style> <style lang="scss">
@use './styles/animations';
// APP
#app {
color: white;
overflow-x: hidden;
font-size: 1em;
}
// CONTAINER
.app_container {
display: grid;
grid-template-rows: auto 1fr auto;
grid-template-columns: 100%;
min-height: 100vh;
overflow: hidden;
position: relative;
}
.app_main {
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>
+39
View File
@@ -0,0 +1,39 @@
<template>
<footer class="app_footer">
&copy;
<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="" />&nbsp;<b>{{ $t('footer.discord') }}</b>
</a>
<div style="display: none">&int; 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>
+23 -21
View File
@@ -18,7 +18,12 @@
<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>
@@ -29,11 +34,6 @@
<img src="/images/icon-dispatcher.svg" alt="icon dispatcher" /> <img src="/images/icon-dispatcher.svg" alt="icon dispatcher" />
<span class="text--primary">{{ onlineDispatchersCount }}</span> <span class="text--primary">{{ onlineDispatchersCount }}</span>
<!-- <span class="g-tooltip">
<b class="text--primary">{{ factorU }}U</b>
<div class="content">Test</div>
</span> -->
<span class="text--grayed"> / </span> <span class="text--grayed"> / </span>
<span class="text--primary">{{ onlineTrainsCount }}</span> <span class="text--primary">{{ onlineTrainsCount }}</span>
<img src="/images/icon-train.svg" alt="icon train" /> <img src="/images/icon-train.svg" alt="icon train" />
@@ -45,17 +45,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"
> >
@@ -74,7 +74,10 @@ import Clock from './Clock.vue';
import RegionDropdown from '../Global/RegionDropdown.vue'; import RegionDropdown from '../Global/RegionDropdown.vue';
export default defineComponent({ export default defineComponent({
components: { StatusIndicator, Clock, RegionDropdown },
emits: ['changeLang'], emits: ['changeLang'],
props: { props: {
currentLang: { currentLang: {
type: String, type: String,
@@ -105,18 +108,17 @@ export default defineComponent({
).length; ).length;
}, },
factorU() { isChristmas() {
return this.onlineDispatchersCount == 0 const date = new Date();
? '-'
: (this.onlineTrainsCount / this.onlineDispatchersCount).toFixed(2); return date.getUTCMonth() == 11 && date.getUTCDate() >= 20 && date.getUTCDate() <= 31;
} }
}, }
components: { StatusIndicator, Clock, RegionDropdown }
}); });
</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 {
@@ -124,7 +126,7 @@ export default defineComponent({
justify-content: center; justify-content: center;
position: relative; position: relative;
background-color: $primaryCol; background-color: #2c2c2c;
} }
.header { .header {
@@ -139,7 +141,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;
} }
@@ -178,7 +180,7 @@ export default defineComponent({
padding: 0.5em; padding: 0.5em;
@include smallScreen { @include responsive.smallScreen{
transform: translateX(85%); transform: translateX(85%);
} }
} }
+1 -5
View File
@@ -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;
+3 -3
View File
@@ -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;
} }
} }
+121
View File
@@ -0,0 +1,121 @@
<template>
<Card :is-open="isUpdateCardOpen" @toggle-card="toggleCard(false)">
<div class="content">
<h1 style="margin-bottom: 0.5em">🚀 {{ $t('update.title') }}</h1>
<div class="features-body" v-if="htmlChangelog != ''" v-html="htmlChangelog"></div>
<div class="no-features" v-else>{{ $t('update.no-data') }}</div>
<button class="btn btn--action" ref="confirm-btn" @click="toggleCard(false)">
{{ $t('update.confirm') }}
</button>
<p class="bottom-info">
{{ $t('update.info-1') }}
<br />
<span v-html="$t('update.info-2')"></span>
</p>
</div>
</Card>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import { useMainStore } from '../../store/mainStore';
import { version } from '../../../package.json';
import { Converter } from 'showdown';
import Card from '../Global/Card.vue';
const converter = new Converter();
export default defineComponent({
components: { Card },
props: {
isUpdateCardOpen: {
type: Boolean,
required: true
}
},
emits: ['toggleCard'],
data() {
return {
mainStore: useMainStore(),
version: version
};
},
watch: {
isUpdateCardOpen(val: boolean) {
this.$nextTick(() => {
if (val) (this.$refs['confirm-btn'] as HTMLElement).focus();
});
}
},
computed: {
htmlChangelog() {
if (this.mainStore.appUpdate == null) return '';
return converter.makeHtml(this.mainStore.appUpdate.changelog);
}
},
methods: {
toggleCard(value: boolean) {
this.$emit('toggleCard', value);
}
}
});
</script>
<style lang="scss" scoped>
::v-deep(h1) {
text-align: center;
color: var(--clr-primary);
}
::v-deep(h2) {
padding: 0.25em 0;
border-bottom: 1px solid #aaa;
}
::v-deep(ul) {
list-style: initial;
padding: 1em;
line-height: 1.5em;
}
.content {
display: grid;
grid-template-rows: auto 1fr auto;
gap: 0.5em;
padding: 1em;
min-height: 700px;
overflow: auto;
text-align: justify;
max-width: 700px;
}
.no-features {
text-align: center;
}
button {
margin: 0 auto;
padding: 0.5em 0.75em;
font-size: 1.1em;
}
p.bottom-info {
text-align: center;
color: #ccc;
}
a {
text-decoration: underline;
}
</style>
-48
View File
@@ -1,48 +0,0 @@
<template>
<AnimatedModal :is-open="mainStore.isNewUpdate" @toggle-modal="toggleModal">
<div class="modal_content">
<h1 class="header">Aktualizacja Stacjownika</h1>
<h2>wersja {{ version }}</h2>
<b>Co nowego?</b>
<p>
<ul>
<li>test</li>
</ul>
</p>
</div>
</AnimatedModal>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import { useMainStore } from '../../store/mainStore';
import { version } from '../../../package.json';
import AnimatedModal from '../Global/AnimatedModal.vue';
export default defineComponent({
components: { AnimatedModal },
data() {
return {
mainStore: useMainStore(),
version: version
};
},
methods: {
toggleModal(value: boolean) {
this.$emit('toggleModal', value);
}
}
});
</script>
<style lang="scss" scoped>
.modal_content {
text-align: center;
padding: 1em;
height: 80vh;
min-height: 550px;
}
</style>
+1 -2
View File
@@ -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;
-2
View File
@@ -38,5 +38,3 @@ export default defineComponent({
} }
}); });
</script> </script>
<style scoped></style>
-101
View File
@@ -1,101 +0,0 @@
<template>
<transition name="modal-anim" tag="div" class="modal">
<div class="body" v-if="isOpen">
<div class="background" @click="toggleModal(false)"></div>
<div class="wrapper" ref="wrapper" tabindex="0">
<slot></slot>
</div>
<div class="tab-exit" ref="exit" tabindex="0" @focus="toggleModal(false)"></div>
</div>
</transition>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import { useMainStore } from '../../store/mainStore';
export default defineComponent({
emits: ['toggleModal'],
props: {
isOpen: Boolean
},
data() {
return {
store: useMainStore()
};
},
watch: {
isOpen(v) {
this.$nextTick(() => {
if (v) (this.$refs['wrapper'] as HTMLElement).focus();
else (this.store.modalLastClickedTarget as HTMLElement)?.focus();
});
}
},
methods: {
toggleModal(value: boolean) {
this.$emit('toggleModal', value);
}
}
});
</script>
<style lang="scss" scoped>
@import '../../styles/responsive.scss';
.body {
position: fixed;
top: 0;
left: 0;
z-index: 200;
width: 100vw;
height: 100vh;
}
.background {
position: absolute;
top: 0;
left: 0;
width: 100vw;
height: 100vh;
cursor: pointer;
background-color: rgba(0, 0, 0, 0.55);
}
.wrapper {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
background-color: #1a1a1a;
box-shadow: 0 0 15px 10px #333333;
width: 95%;
max-width: 800px;
max-height: 95vh;
& > :slotted(div) {
max-height: 95vh;
}
}
@include smallScreen {
.wrapper {
top: 0;
transform: translate(-50%, 1em);
max-height: 90vh;
& > :slotted(div) {
max-height: 90vh;
}
}
}
</style>
+93
View File
@@ -0,0 +1,93 @@
<template>
<transition name="modal-anim" tag="div">
<div class="card" v-if="isOpen">
<div class="card-background" @click="toggleCard(false)"></div>
<div class="card-body" tabindex="0">
<slot></slot>
</div>
</div>
</transition>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import { useMainStore } from '../../store/mainStore';
export default defineComponent({
emits: ['toggleCard'],
props: {
isOpen: Boolean
},
data() {
return {
store: useMainStore()
};
},
watch: {
isOpen(v) {
this.$nextTick(() => {
if (v == false) (this.store.modalLastClickedTarget as HTMLElement)?.focus();
});
}
},
methods: {
toggleCard(value: boolean) {
this.$emit('toggleCard', value);
}
}
});
</script>
<style lang="scss" scoped>
@use '../../styles/responsive';
.card {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
z-index: 200;
display: flex;
justify-content: center;
align-items: center;
}
.card-background {
position: absolute;
top: 0;
left: 0;
width: 100vw;
height: 100vh;
cursor: pointer;
background-color: rgba(0, 0, 0, 0.55);
}
.card-body {
position: relative;
margin: 1em;
max-height: 95vh;
max-height: 95dvh;
background-color: #1a1a1a;
box-shadow: 0 0 15px 10px #0e0e0e;
overflow: auto;
}
@include responsive.smallScreen{
.card {
align-items: flex-start;
}
}
</style>
@@ -1,12 +1,7 @@
<template> <template>
<AnimatedModal <Card :isOpen="isCardOpen" @toggleCard="toggleCard" @keydown.esc="toggleCard(false)">
class="donation-modal" <div class="body">
:isOpen="isModalOpen" <div class="content">
@toggleModal="toggleModal"
@keydown.esc="toggleModal(false)"
>
<div class="modal_content">
<div class="modal_main">
<h1 v-html="$t('donations.header')"></h1> <h1 v-html="$t('donations.header')"></h1>
<div class="donators-slider" v-if="donatorList.length != 0"> <div class="donators-slider" v-if="donatorList.length != 0">
<span v-html="$t('donations.donator-title', { count: donatorList.length })"></span> <span v-html="$t('donations.donator-title', { count: donatorList.length })"></span>
@@ -61,18 +56,19 @@
</i> </i>
</div> </div>
<div class="modal_actions"> <div class="actions-container">
<a <a
class="modal-action a-button btn--image coffee" class="action a-button btn--image coffee"
href="https://buycoffee.to/spythere" href="https://buycoffee.to/spythere"
target="_blank" target="_blank"
ref="action"
> >
<img src="/images/icon-coffee.png" width="20" alt="buycoffee.to donation" /> <img src="/images/icon-coffee.png" width="20" alt="buycoffee.to donation" />
{{ $t('donations.action-buycoffee') }} {{ $t('donations.action-buycoffee') }}
</a> </a>
<a <a
class="modal-action a-button btn--image paypal" class="action a-button btn--image paypal"
href="https://www.paypal.com/donate/?hosted_button_id=EDB3SKFAHXFTW" href="https://www.paypal.com/donate/?hosted_button_id=EDB3SKFAHXFTW"
target="_blank" target="_blank"
> >
@@ -80,32 +76,36 @@
{{ $t('donations.action-paypal') }} {{ $t('donations.action-paypal') }}
</a> </a>
<button class="modal-action btn--image exit" @click="toggleModal(false)"> <button class="action btn--image exit" @click="toggleCard(false)">
<img src="/images/icon-exit.svg" alt="dollar donation icon" /> <img src="/images/icon-exit.svg" alt="dollar donation icon" />
{{ $t('donations.action-exit') }} {{ $t('donations.action-exit') }}
</button> </button>
</div> </div>
</div> </div>
</AnimatedModal> </Card>
</template> </template>
<script lang="ts"> <script lang="ts">
import { defineComponent } from 'vue'; import { defineComponent } from 'vue';
import AnimatedModal from './AnimatedModal.vue';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
import Card from './Card.vue';
export default defineComponent({ export default defineComponent({
components: { AnimatedModal }, components: { Card },
props: { props: {
isModalOpen: Boolean isCardOpen: Boolean
}, },
emits: ['toggleModal'], emits: ['toggleCard'],
watch: { watch: {
isModalOpen(b: boolean) { isCardOpen(val: boolean) {
this.running = b; this.running = val;
this.lastUpdate = Date.now(); this.lastUpdate = Date.now();
this.$nextTick(() => {
if (val) (this.$refs['action'] as HTMLElement).focus();
});
} }
}, },
@@ -133,8 +133,8 @@ export default defineComponent({
}, },
methods: { methods: {
toggleModal(value: boolean) { toggleCard(value: boolean) {
this.$emit('toggleModal', value); this.$emit('toggleCard', value);
}, },
runUpdate() { runUpdate() {
@@ -150,80 +150,74 @@ export default defineComponent({
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@import '../../styles/responsive.scss'; .body {
.modal_content {
display: grid; display: grid;
grid-template-rows: 1fr auto; grid-template-rows: 1fr auto;
gap: 1em; gap: 1em;
font-size: 1.1em; font-size: 1.1em;
& > div { max-width: 820px;
padding: 1em;
}
h1 {
font-size: 1.95em;
text-align: center;
}
p {
text-align: justify;
}
a.discord {
text-decoration: underline;
}
} }
.modal_main { .content {
overflow: auto; overflow: auto;
overflow-x: hidden; overflow-x: hidden;
padding: 1em;
img {
max-height: 20px;
margin-right: 5px;
vertical-align: text-bottom;
}
} }
.modal_actions { img {
max-height: 20px;
margin-right: 5px;
vertical-align: text-bottom;
}
h1 {
font-size: 1.95em;
text-align: center;
}
p {
text-align: justify;
}
a.discord {
text-decoration: underline;
}
.actions-container {
display: grid; display: grid;
grid-template-columns: repeat(auto-fit, minmax(220px, 1fr)); grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 0.5em; gap: 0.5em;
padding: 1em;
form button { form button {
width: 100%; width: 100%;
} }
} }
.modal_actions > .modal-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;
} }
} }
} }
-1
View File
@@ -43,7 +43,6 @@ export default defineComponent({
width: 6em; width: 6em;
height: 1em; height: 1em;
margin: 0.5em 0;
.bar-fg, .bar-fg,
.bar-bg { .bar-bg {
+6 -15
View File
@@ -65,12 +65,12 @@ export default defineComponent({
immediate: true, immediate: true,
handler(regionQuery: string) { handler(regionQuery: string) {
if (regionQuery) { if (regionQuery) {
this.store.region.id = this.store.region =
regionsJSON.find( regionsJSON.find(
(reg) => (reg) =>
reg.id == regionQuery.toLocaleLowerCase() || reg.id == regionQuery.toLocaleLowerCase() ||
reg.value.toLocaleLowerCase() == regionQuery.toLocaleLowerCase() reg.value.toLocaleLowerCase() == regionQuery.toLocaleLowerCase()
)?.id || 'eu'; ) ?? regionsJSON[0];
} }
} }
} }
@@ -84,7 +84,7 @@ export default defineComponent({
regionList() { regionList() {
return regionsJSON.map((region) => { return regionsJSON.map((region) => {
const regionStationCount = this.store.activeSceneryList.filter( const regionStationCount = this.store.activeSceneryList.filter(
(scenery) => scenery.region == region.id (scenery) => scenery.region == region.id && scenery.dispatcherId != -1
).length; ).length;
const regionTrainCount = const regionTrainCount =
@@ -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;
@@ -139,15 +137,10 @@ button.selected-region {
color: paleturquoise; color: paleturquoise;
font-weight: bold; font-weight: bold;
padding: 0.1em 0.5em;
&:focus { &:focus {
background-color: #262626; background-color: #262626;
} }
span {
margin-right: 10px;
}
} }
.content { .content {
@@ -187,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;
} }
} }
@@ -197,6 +190,8 @@ li.option {
} }
label { label {
width: 100%;
padding: 0.5em 0;
position: relative; position: relative;
display: inline-block; display: inline-block;
@@ -207,10 +202,6 @@ li.option {
background-color: #333333f2; background-color: #333333f2;
} }
padding: 0.5em 0;
width: 100%;
cursor: pointer; cursor: pointer;
} }
} }
+13 -23
View File
@@ -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%;
} }
} }
+8 -13
View File
@@ -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],
@@ -88,8 +82,9 @@ $unknown: #b93c3c;
.status-badge { .status-badge {
border-radius: 1em; border-radius: 1em;
font-weight: 500; font-weight: 500;
text-wrap: nowrap;
padding: 0.2em 0.55em; padding: 0.2rem 0.55rem;
background-color: $online; background-color: $online;
@@ -106,13 +101,13 @@ $unknown: #b93c3c;
&.no-limit { &.no-limit {
background-color: $no-limit; background-color: $no-limit;
font-size: 0.85em; font-size: 0.9em;
} }
&.not-signed, &.not-signed,
&.unavailable { &.unavailable {
background-color: $unav; background-color: $unav;
font-size: 0.85em; font-size: 0.9em;
} }
&.afk { &.afk {
@@ -125,7 +120,7 @@ $unknown: #b93c3c;
background-color: $no-space; background-color: $no-space;
border: 1px solid white; border: 1px solid white;
color: white; color: white;
font-size: 0.85em; font-size: 0.9em;
} }
&.unknown, &.unknown,
+116 -115
View File
@@ -1,66 +1,13 @@
<template> <template>
<div class="stock-list"> <div class="list-wrapper">
<div v-if="tractionOnly"> <ul class="stock-list">
<p> <li v-for="({ images, imagesFallbacks, vehicleString }, i) in thumbnailNames">
{{ computedStockList[0].split(':')[0].split('_').splice(0, 2).join(' ') }} <VehicleThumbnail
{{ computedStockList[0].split(':')[1] }} :key="i"
</p> :vehicle-string="vehicleString"
:images="images"
<img :image-fallbacks="imagesFallbacks"
:src="`https://rj.td2.info.pl/dist/img/thumbnails/${computedStockList[0].split(':')[0]}${ />
/^EN/.test(computedStockList[0]) ? 'rb' : ''
}.png`"
@error="onImageError($event, computedStockList[0])"
width="400"
height="60"
/>
</div>
<ul v-else>
<li v-for="(stockName, i) in computedStockList" :key="i">
<p>
{{ stockName.split(':')[0].split('_').splice(0, 2).join(' ') }}
{{ stockName.split(':')[1] }}
</p>
<span>
<img
:src="`https://rj.td2.info.pl/dist/img/thumbnails/${stockName.split(':')[0]}${
/^EN/.test(stockName) ? 'rb' : ''
}.png`"
@error="onImageError($event, stockName)"
width="400"
height="60"
/>
<!-- /// Manualne dodawanie miniaturek członów dla kibelków /// -->
<img
v-if="/^(EN|2EN)/.test(stockName)"
:src="`https://rj.td2.info.pl/dist/img/thumbnails/${stockName.split(':')[0]}s.png`"
@error="
(event) => ((event.target as HTMLImageElement).src = '/images/icon-loco-ezt-s.png')
"
/>
<img
class="train-thumbnail"
v-if="/^EN71/.test(stockName)"
:src="`https://rj.td2.info.pl/dist/img/thumbnails/${stockName.split(':')[0]}s.png`"
@error="
(event) => ((event.target as HTMLImageElement).src = '/images/icon-loco-ezt-s.png')
"
/>
<img
class="train-thumbnail"
v-if="/^(EN|2EN)/.test(stockName)"
:src="`https://rj.td2.info.pl/dist/img/thumbnails/${stockName.split(':')[0]}ra.png`"
@error="
(event) => ((event.target as HTMLImageElement).src = '/images/icon-loco-ezt-ra.png')
"
/>
<!-- /// -->
</span>
</li> </li>
</ul> </ul>
</div> </div>
@@ -69,88 +16,142 @@
<script lang="ts"> <script lang="ts">
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
import VehicleThumbnail from './VehicleThumbnail.vue';
export default defineComponent({ export default defineComponent({
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: {
computedStockList() { computedStockList() {
return this.tractionOnly ? this.trainStockList.slice(0, 1) : this.trainStockList; return this.tractionOnly ? this.trainStockList.slice(0, 1) : this.trainStockList;
} },
},
methods: { thumbnailNames() {
onImageError(event: Event, stockName: string) { return (this.tractionOnly ? this.trainStockList.slice(0, 1) : this.trainStockList)
let fallbackName = ''; .filter((v) => v.length != 0)
.map((vehicleString) => {
const [vehicleName] = vehicleString.split(':');
const isLoco = /.-\d/.test(stockName); const vehicleThumbnailData = {
images: [] as string[],
imagesFallbacks: [] as string[],
vehicleName,
vehicleString
};
if (isLoco) { // Generowanie członów EN57
fallbackName += 'loco-'; if (vehicleName.startsWith('EN57')) {
fallbackName += /^\d?EN\d{2}/.test(stockName) vehicleThumbnailData['images'] = [
? 'ezt' vehicleName + 'ra',
: /^SN\d{2}/.test(stockName) vehicleName + 's',
? 'szt' vehicleName + 'rb'
: /^\d?E/.test(stockName) ];
? 'e' vehicleThumbnailData['imagesFallbacks'] = [
: 's'; 'unknown_ezt-ra',
} else { 'unknown_ezt-s',
const isCarPassenger = /(\d{3}a|(Bau|Gor)\d{2}|304C)_/.test(stockName); 'unknown_ezt-rb'
];
}
// Generowanie członów EN71
else if (vehicleName.startsWith('EN71')) {
vehicleThumbnailData['images'] = [
vehicleName + 'ra',
vehicleName + 'sa',
vehicleName + 'sb',
vehicleName + 'rb'
];
vehicleThumbnailData['imagesFallbacks'] = [
'unknown_ezt-ra',
'unknown_ezt-sa',
'unknown_ezt-sb',
'unknown_ezt-rb'
];
}
// Generowanie pojazdów i członów 2EN57
else if (vehicleString.startsWith('2EN57')) {
const [firstVehicleNumber, secondVehicleNumber] = vehicleString
.replace('2EN57-', '')
.split('+');
fallbackName += 'car-'; vehicleThumbnailData['images'] = [
fallbackName += isCarPassenger ? 'passenger' : 'cargo'; `EN57-${firstVehicleNumber}ra`,
} `EN57-${firstVehicleNumber}s`,
`EN57-${firstVehicleNumber}rb`,
`EN57-${secondVehicleNumber}ra`,
`EN57-${secondVehicleNumber}s`,
`EN57-${secondVehicleNumber}rb`
];
(event.target as HTMLImageElement).src = `/images/icon-${fallbackName}.png`; vehicleThumbnailData['imagesFallbacks'] = [
'unknown_ezt-ra',
'unknown_ezt-s',
'unknown_ezt-rb',
'unknown_ezt-ra',
'unknown_ezt-s',
'unknown_ezt-rb'
];
}
// Generowanie członów Gor77
else if (vehicleString.startsWith('Gor77')) {
vehicleThumbnailData['images'] = [
vehicleName + '-A',
vehicleName + '-B',
vehicleName + '-C',
vehicleName + '-D'
];
vehicleThumbnailData['imagesFallbacks'] = [
'unknown_Gor77-A',
'unknown_Gor77-B',
'unknown_Gor77-C',
'unknown_Gor77-D'
];
}
// Generowanie członów ET41
else if (vehicleString.startsWith('ET41')) {
vehicleThumbnailData['images'] = [vehicleName + '-A', vehicleName + '-B'];
vehicleThumbnailData['imagesFallbacks'] = ['unknown_ET41-A', 'unknown_ET41-B'];
}
// Generowanie pozostałych pojazdów
else {
let fallbackVehicleImage = 'unknown_cargo';
if (/^(EP|EU|ET|201E)/.test(vehicleName)) fallbackVehicleImage = 'unknown_train';
else if (/^(SM42)/.test(vehicleName)) fallbackVehicleImage = 'unknown_SM42';
else if (/(\d{3}a|(Bau|Gor)\d{2}|304C)_/.test(vehicleName))
fallbackVehicleImage = 'unknown_passenger';
vehicleThumbnailData['images'] = [vehicleName];
vehicleThumbnailData['imagesFallbacks'] = [fallbackVehicleImage];
}
if (this.tractionOnly) vehicleThumbnailData['images'].length = 1;
return vehicleThumbnailData;
});
} }
} }
}); });
</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;
padding: 1em 0;
}
ul > li > span {
display: flex;
align-items: flex-end;
}
img {
max-height: 60px;
width: auto;
height: auto;
}
p {
text-align: center;
color: #aaa;
font-size: 0.95em;
margin-bottom: 1em;
} }
</style> </style>
@@ -0,0 +1,80 @@
<template>
<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
v-for="(thumbnailImage, imageIndex) in images"
:src="`https://stacjownik.spythere.eu/static/thumbnails/${thumbnailImage}.png`"
height="60"
loading="lazy"
data-tooltip-type="VehiclePreviewTooltip"
:data-tooltip-content="vehicleString"
@error="onImageError($event, imageFallbacks[imageIndex])"
@load="onImageLoad"
/>
</div>
</div>
</template>
<script setup lang="ts">
import { computed, PropType, Ref, ref } from 'vue';
const props = defineProps({
vehicleString: { type: String, required: true },
images: { type: Object as PropType<string[]>, required: true },
imageFallbacks: { type: Object as PropType<string[]>, required: true }
});
const thumbRef = ref(null) as Ref<HTMLElement | null>;
const imgStatus = ref('loading');
const vehicleName = computed(() => props.vehicleString.split(':')[0].replace(/_/g, ' '));
const vehicleCargo = computed(() => props.vehicleString.split(':')[1]);
function onImageError(event: Event, fallbackImage: string) {
(event.target as HTMLImageElement).src = `/images/${fallbackImage}.png`;
imgStatus.value = 'error';
}
function onImageLoad() {
if (imgStatus.value != 'error') {
imgStatus.value = 'loaded';
}
if (thumbRef.value) thumbRef.value.style.opacity = '1';
}
</script>
<style lang="scss" scoped>
.vehicle-thumbnail {
position: relative;
opacity: 0;
transition: opacity 100ms ease-in-out;
&[data-load-status='loading'] {
min-height: 60px;
min-width: 200px;
}
}
.stock-text {
text-align: center;
color: #aaa;
font-size: 0.9em;
margin-bottom: 0.25em;
padding: 0.25em 0;
}
.stock-images {
display: flex;
justify-content: center;
align-items: flex-end;
cursor: crosshair;
padding: 0.5em 0;
}
</style>
@@ -1,29 +1,28 @@
<template> <template>
<section class="daily-stats"> <section class="daily-stats">
<span :data-active="statsStatus"> <span :data-active="statsStatus">
<span class="stats-list"> <h3>
<h3> {{ $t('journal.daily-stats.title') }}
{{ $t('journal.daily-stats.title') }} <b class="text--primary">{{ new Date().toLocaleDateString($i18n.locale) }}</b>
<b class="text--primary">{{ new Date().toLocaleDateString($i18n.locale) }}</b> </h3>
</h3>
<hr class="header-separator" /> <hr class="header-separator" />
<b v-if="statsStatus == Status.Data.Loading"> <b v-if="statsStatus == Status.Data.Loading">
{{ $t('app.loading') }} {{ $t('app.loading') }}
</b> </b>
<b class="text--error" v-else-if="statsStatus == Status.Data.Error"> <b class="text--error" v-else-if="statsStatus == Status.Data.Error">
{{ $t('journal.stats-error') }} {{ $t('journal.stats-error') }}
</b> </b>
<b v-else-if="topDispatchers.length == 0"> <b v-else-if="topDispatchers.length == 0">
{{ $t('journal.daily-stats.info') }} {{ $t('journal.daily-stats.info') }}
</b> </b>
<div v-else> <div v-else>
<div v-if="stats.totalTimetables"> <ul class="stats-list">
&bull; <li v-if="stats.totalTimetables">
<i18n-t keypath="journal.daily-stats.total"> <i18n-t keypath="journal.daily-stats.total">
<template #count> <template #count>
<b class="text--primary"> <b class="text--primary">
@@ -36,10 +35,9 @@
<b class="text--primary"> {{ stats.distanceSum?.toFixed(2) }} km</b> <b class="text--primary"> {{ stats.distanceSum?.toFixed(2) }} km</b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="stats.maxTimetable"> <li v-if="stats.maxTimetable">
&bull;
<i18n-t keypath="journal.daily-stats.longest"> <i18n-t keypath="journal.daily-stats.longest">
<template #id> <template #id>
<router-link :to="`/journal/timetables?search-train=%23${stats.maxTimetable.id}`"> <router-link :to="`/journal/timetables?search-train=%23${stats.maxTimetable.id}`">
@@ -60,10 +58,9 @@
<b class="text--primary">{{ stats.maxTimetable.routeDistance }} km</b> <b class="text--primary">{{ stats.maxTimetable.routeDistance }} km</b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="topDispatchers.length == 1"> <li v-if="topDispatchers.length == 1">
&bull;
<i18n-t keypath="journal.daily-stats.most-active-dr"> <i18n-t keypath="journal.daily-stats.most-active-dr">
<template #dispatcher> <template #dispatcher>
<router-link <router-link
@@ -79,10 +76,9 @@
</b> </b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="topDispatchers.length > 1"> <li v-if="topDispatchers.length > 1">
&bull;
<i18n-t keypath="journal.daily-stats.most-active-dr-many"> <i18n-t keypath="journal.daily-stats.most-active-dr-many">
<template #dispatchers> <template #dispatchers>
<span v-for="(disp, i) in topDispatchers" :key="i"> <span v-for="(disp, i) in topDispatchers" :key="i">
@@ -103,10 +99,9 @@
</b> </b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="stats.longestDuties.length > 0"> <li v-if="stats.longestDuties.length > 0">
&bull;
<i18n-t keypath="journal.daily-stats.longest-duties"> <i18n-t keypath="journal.daily-stats.longest-duties">
<template #dispatcher> <template #dispatcher>
<router-link <router-link
@@ -122,10 +117,9 @@
{{ calculateDuration(stats.longestDuties[0].duration) }} {{ calculateDuration(stats.longestDuties[0].duration) }}
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
<div v-if="stats.mostActiveDrivers.length > 0"> <li v-if="stats.mostActiveDrivers.length > 0">
&bull;
<i18n-t keypath="journal.daily-stats.most-active-driver"> <i18n-t keypath="journal.daily-stats.most-active-driver">
<template #driver> <template #driver>
<router-link <router-link
@@ -138,30 +132,30 @@
<b class="text--primary">{{ stats.mostActiveDrivers[0].distance.toFixed(2) }} km</b> <b class="text--primary">{{ stats.mostActiveDrivers[0].distance.toFixed(2) }} km</b>
</template> </template>
</i18n-t> </i18n-t>
</div> </li>
</ul>
<hr class="section-separator" /> <hr class="section-separator" />
<div class="stats-badges"> <div class="stats-badges">
<span <span
class="stat-badge" class="badge stat-badge"
v-for="key in [ v-for="key in [
'rippedSwitches', 'rippedSwitches',
'derailments', 'derailments',
'skippedStopSignals', 'skippedStopSignals',
'radioStops', 'radioStops',
'kills' 'kills'
]" ]"
:key="key" :key="key"
> >
<span>{{ $t(`journal.daily-stats.${key}`) }}</span> <span>{{ $t(`journal.daily-stats.${key}`) }}</span>
<span>{{ <span>
Object.entries(stats.globalDiff).find(([k, v]) => k == key)?.[1] || '--' {{ Object.entries(stats.globalDiff).find(([k, v]) => k == key)?.[1] || '--' }}
}}</span>
</span> </span>
</div> </span>
</div> </div>
</span> </div>
</span> </span>
</section> </section>
</template> </template>
@@ -178,7 +172,6 @@ export default defineComponent({
name: 'journal-daily-stats', name: 'journal-daily-stats',
mixins: [dateMixin], mixins: [dateMixin],
// emits: ['toggleStatsOpen'],
data() { data() {
return { return {
@@ -193,7 +186,6 @@ export default defineComponent({
activated() { activated() {
this.startFetchingDailyStats(); this.startFetchingDailyStats();
// this.$emit('toggleStatsOpen', true);
}, },
deactivated() { deactivated() {
@@ -242,28 +234,38 @@ 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;
} }
.daily-stats > span[data-active='0'] { .daily-stats > span[data-active='0'] {
opacity: 0.75; opacity: 0.75;
} }
ul.stats-list {
list-style: disc;
padding: 0 1em;
}
.stats-list a { .stats-list a {
text-decoration: underline; text-decoration: underline;
} }
.stats-list > li {
margin: 0.25em 0;
}
.stats-badges { .stats-badges {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
gap: 0.5em; gap: 0.5em;
} }
@include smallScreen { @include responsive.smallScreen{
h3 { h3 {
text-align: center; text-align: center;
} }
@@ -0,0 +1,208 @@
<template>
<li class="dispatcher-history-entry">
<div class="entry-info">
<span>
<span>
<router-link :to="`/journal/dispatchers?search-station=${entry.stationName}`">
<b>{{ entry.stationName }}</b>
</router-link>
<b class="text--grayed"> #{{ entry.stationHash }}</b>
</span>
&bull;
<b
v-if="entry.dispatcherLevel !== null"
class="level-badge dispatcher"
:style="calculateExpStyle(entry.dispatcherLevel, entry.dispatcherIsSupporter)"
>
{{ entry.dispatcherLevel >= 2 ? entry.dispatcherLevel : 'L' }}
</b>
<b style="margin-left: 5px">
<span
v-if="apiStore.donatorsData.includes(entry.dispatcherName)"
data-tooltip-type="DonatorTooltip"
:data-tooltip-content="$t('donations.dispatcher-message')"
>
<router-link
class="text--donator"
:to="`/journal/dispatchers?search-dispatcher=${entry.dispatcherName}`"
>
{{ entry.dispatcherName }}
</router-link>
</span>
<router-link
v-else
:to="`/journal/dispatchers?search-dispatcher=${entry.dispatcherName}`"
>
{{ entry.dispatcherName }}
</router-link>
</b>
<div>
<span v-if="entry.timestampTo">
<b>{{ $d(entry.timestampFrom) }}</b>
{{ timestampToString(entry.timestampFrom) }}
-
<b
v-if="
new Date(entry.timestampFrom).getDate() != new Date(entry.timestampTo).getDate()
"
>
{{ $d(entry.timestampTo) }}
</b>
{{ timestampToString(entry.timestampTo) }} ({{
calculateDuration(entry.currentDuration)
}})
</span>
<router-link
:to="`/scenery?station=${entry.stationName}`"
class="dispatcher-online"
v-else
>
{{ $t('journal.online-since') }}
<b>
{{
new Date().getDate() != new Date(entry.timestampFrom).getDate()
? $d(entry.timestampFrom)
: ''
}}
{{ timestampToString(entry.timestampFrom) }}
</b>
({{ calculateDuration(entry.currentDuration) }})
</router-link>
</div>
</span>
<span class="entry-info-right">
<div>
<span>
{{ $t('scenery.dispatcher-rate') }}
<b class="text--primary"> {{ entry.dispatcherRate }}</b>
</span>
<button class="btn btn--option" @click="toggleExtraInfo">
{{ $t('scenery.dispatcher-status-changes') }}
<b class="text--primary">{{ entry.statusHistory.length }}</b>
</button>
</div>
<b class="region-badge" :aria-describedby="entry.region">
REGION: {{ regions.find((r) => r.id == entry.region)?.name }}
</b>
</span>
</div>
<div class="entry-extra" v-if="showExtraInfo">
<ul class="status-list">
<li v-for="statusItem in entry.statusHistory">
<b style="margin-right: 0.5em">{{
timestampToString(parseInt(statusItem.split('@')[0]))
}}</b>
<StationStatusBadge
:dispatcher-status="Number(statusItem.split('@')[1])"
:is-online="true"
/>
</li>
</ul>
</div>
</li>
</template>
<script lang="ts">
import { defineComponent, PropType } from 'vue';
import { regions } from '../../../data/options.json';
import { API } from '../../../typings/api';
import dateMixin from '../../../mixins/dateMixin';
import styleMixin from '../../../mixins/styleMixin';
import { useApiStore } from '../../../store/apiStore';
import StationStatusBadge from '../../Global/StationStatusBadge.vue';
export default defineComponent({
props: {
entry: { type: Object as PropType<API.DispatcherHistory.Data>, required: true },
showExtraInfo: { type: Boolean, required: true }
},
components: { StationStatusBadge },
mixins: [dateMixin, styleMixin],
emits: ['toggleShowExtraInfo'],
data() {
return { regions, apiStore: useApiStore() };
},
methods: {
toggleExtraInfo() {
this.$emit('toggleShowExtraInfo', this.entry.id);
}
}
});
</script>
<style lang="scss" scoped>
@use '../../../styles/responsive';
@use '../../../styles/badge';
.region-badge {
padding: 0 0.25em;
}
.level-badge {
text-align: center;
display: inline-block;
line-height: 1.6em;
}
.dispatcher-online {
color: springgreen;
}
.dispatcher-history-entry {
background-color: #1a1a1a;
padding: 1em;
}
.entry-info {
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: wrap;
line-height: 1.75em;
gap: 0.5em;
}
.entry-info-right {
display: flex;
flex-wrap: wrap;
align-items: center;
text-align: center;
gap: 1em;
}
.entry-extra {
margin-top: 1em;
}
.status-list {
display: flex;
overflow: auto;
gap: 0.5em;
}
.status-list > li {
background-color: #313131;
padding: 0.2rem 0 0.2rem 0.5em;
margin: 0.5em 0;
border-radius: 1em;
}
@include responsive.smallScreen{
.entry-info {
flex-direction: column;
justify-content: center;
text-align: center;
}
}
</style>
@@ -16,41 +16,41 @@
<hr class="header-separator" /> <hr class="header-separator" />
<div class="info-stats"> <div class="info-stats">
<span class="stat-badge" v-if="stats.services"> <span class="badge stat-badge" v-if="stats.services">
<span>{{ $t('journal.dispatcher-stats.services-count') }}</span> <span>{{ $t('journal.dispatcher-stats.services-count') }}</span>
<span>{{ stats.services.count }}</span> <span>{{ stats.services.count }}</span>
</span> </span>
<span class="stat-badge" v-if="stats.services"> <span class="badge stat-badge" v-if="stats.services">
<span>{{ $t('journal.dispatcher-stats.service-max') }}</span> <span>{{ $t('journal.dispatcher-stats.service-max') }}</span>
<span>{{ calculateDuration(stats.services.durationMax) }}</span> <span>{{ calculateDuration(stats.services.durationMax) }}</span>
</span> </span>
<span class="stat-badge" v-if="stats.services"> <span class="badge stat-badge" v-if="stats.services">
<span>{{ $t('journal.dispatcher-stats.service-avg') }}</span> <span>{{ $t('journal.dispatcher-stats.service-avg') }}</span>
<span>{{ calculateDuration(stats.services.durationAvg) }}</span> <span>{{ calculateDuration(stats.services.durationAvg) }}</span>
</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="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="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="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="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,140 +1,59 @@
<template> <template>
<transition name="status-anim" mode="out-in"> <div>
<div :key="dataStatus"> <div class="journal_warning" v-if="store.isOffline">
<div class="journal_warning" v-if="store.isOffline"> {{ $t('app.offline') }}
{{ $t('app.offline') }}
</div>
<Loading v-else-if="dataStatus == Status.Data.Loading" />
<div v-else-if="dataStatus == Status.Data.Error" class="journal_warning error">
{{ $t('app.error') }}
</div>
<div class="journal_warning" v-else-if="dispatcherHistory.length == 0">
{{ $t('app.no-result') }}
</div>
<div v-else>
<table class="dispatchers-table">
<thead>
<th>{{ $t('journal.history-name') }}</th>
<th>{{ $t('journal.history-hash') }}</th>
<th>{{ $t('journal.history-dispatcher') }}</th>
<th>{{ $t('journal.history-level') }}</th>
<th>{{ $t('journal.history-rate') }}</th>
<th>{{ $t('journal.history-region') }}</th>
<th>{{ $t('journal.history-date') }}</th>
</thead>
<tbody>
<transition-group name="list-anim">
<tr v-for="historyItem in dispatcherHistory" :key="historyItem.id">
<td>
<router-link
:to="`/journal/dispatchers?search-station=${historyItem.stationName}`"
>
<b>{{ historyItem.stationName }}</b>
</router-link>
</td>
<td>#{{ historyItem.stationHash }}</td>
<td>
<router-link
:to="`/journal/dispatchers?search-dispatcher=${historyItem.dispatcherName}`"
>
<b
v-if="isDonator(historyItem.dispatcherName)"
class="text--donator"
:title="$t('donations.dispatcher-message')"
>
{{ historyItem.dispatcherName }}
</b>
<b v-else>
{{ historyItem.dispatcherName }}
</b>
</router-link>
</td>
<td>
<b
v-if="historyItem.dispatcherLevel !== null"
class="level-badge dispatcher"
:style="
calculateExpStyle(
historyItem.dispatcherLevel,
historyItem.dispatcherIsSupporter
)
"
>
{{ historyItem.dispatcherLevel >= 2 ? historyItem.dispatcherLevel : 'L' }}
</b>
</td>
<td class="text--primary">
<b>{{ historyItem.dispatcherRate }}</b>
</td>
<td>
<b class="region-badge" :aria-describedby="historyItem.region">{{
regions.find((r) => r.id == historyItem.region)?.value || '???'
}}</b>
</td>
<td style="min-width: 200px" class="time">
<span v-if="historyItem.timestampTo" class="text--offline">
<b>{{ $d(historyItem.timestampFrom) }}</b>
{{ timestampToString(historyItem.timestampFrom) }}
- {{ timestampToString(historyItem.timestampTo) }} ({{
calculateDuration(historyItem.currentDuration)
}})
</span>
<span class="dispatcher-online" v-else>
<b class="text--online">
<router-link :to="`/scenery?station=${historyItem.stationName}`">{{
$t('journal.online-since')
}}</router-link>
{{ timestampToString(historyItem.timestampFrom) }}
</b>
({{ calculateDuration(historyItem.currentDuration) }})
</span>
</td>
</tr>
</transition-group>
</tbody>
</table>
<AddDataButton
:list="dispatcherHistory"
:scrollDataLoaded="scrollDataLoaded"
:scrollNoMoreData="scrollNoMoreData"
@addHistoryData="addHistoryData"
/>
</div>
<div class="journal_warning" v-if="scrollNoMoreData">
{{ $t('journal.no-further-data') }}
</div>
<div class="journal_warning" v-else-if="!scrollDataLoaded">
{{ $t('journal.loading-further-data') }}
</div>
</div> </div>
</transition>
<Loading v-else-if="dataStatus == Status.Data.Loading" />
<div v-else-if="dataStatus == Status.Data.Error" class="journal_warning error">
{{ $t('app.error') }}
</div>
<div class="journal_warning" v-else-if="dispatcherHistory.length == 0">
{{ $t('app.no-result') }}
</div>
<div v-else>
<transition-group name="list-anim" class="journal-list" tag="ul">
<JournalDispatcherEntry
v-for="entry in dispatcherHistory"
:key="entry.id"
:entry="entry"
:onToggleShowExtraInfo="toggleExtraInfo"
:showExtraInfo="extraInfoIndexes.includes(entry.id)"
/>
</transition-group>
<AddDataButton
:list="dispatcherHistory"
:scrollDataLoaded="scrollDataLoaded"
:scrollNoMoreData="scrollNoMoreData"
@addHistoryData="addHistoryData"
/>
</div>
<div class="journal_warning" v-if="scrollNoMoreData">
{{ $t('journal.no-further-data') }}
</div>
<div class="journal_warning" v-else-if="!scrollDataLoaded">
{{ $t('journal.loading-further-data') }}
</div>
</div>
</template> </template>
<script lang="ts"> <script lang="ts">
import { defineComponent, PropType } from 'vue'; import { defineComponent, PropType } from 'vue';
import { regions } from '../../../data/options.json';
import { useMainStore } from '../../../store/mainStore'; import { useMainStore } from '../../../store/mainStore';
import { API } from '../../../typings/api'; import { API } from '../../../typings/api';
import { Status } from '../../../typings/common'; import { Status } from '../../../typings/common';
import Loading from '../../Global/Loading.vue'; import Loading from '../../Global/Loading.vue';
import AddDataButton from '../../Global/AddDataButton.vue'; import AddDataButton from '../../Global/AddDataButton.vue';
import dateMixin from '../../../mixins/dateMixin'; import JournalDispatcherEntry from './JournalDispatcherEntry.vue';
import donatorMixin from '../../../mixins/donatorMixin';
import styleMixin from '../../../mixins/styleMixin';
export default defineComponent({ export default defineComponent({
components: { Loading, AddDataButton }, components: { Loading, AddDataButton, JournalDispatcherEntry },
mixins: [dateMixin, styleMixin, donatorMixin],
props: { props: {
dispatcherHistory: { dispatcherHistory: {
@@ -159,98 +78,31 @@ export default defineComponent({
return { return {
Status, Status,
store: useMainStore(), store: useMainStore(),
regions
extraInfoIndexes: [] as number[]
}; };
}, },
computed: { watch: {
computedDispatcherHistory() { '$route.query': {
return this.dispatcherHistory.reduce( deep: true,
(acc, historyItem, i) => { handler() {
if (this.isAnotherDay(i - 1, i)) this.extraInfoIndexes.length = 0;
acc.push(new Date(historyItem.timestampFrom).toLocaleDateString('pl-PL')); }
acc.push(historyItem);
return acc;
},
[] as (API.DispatcherHistory.Data | string)[]
);
} }
}, },
methods: { methods: {
navigateToScenery(name: string, isOnline: boolean) { toggleExtraInfo(id: number) {
if (!isOnline) return; const existingIdx = this.extraInfoIndexes.indexOf(id);
this.$router.push(`/scenery?station=${name.trim().replace(/ /g, '_')}`); if (existingIdx != -1) this.extraInfoIndexes.splice(existingIdx, 1);
}, else this.extraInfoIndexes.push(id);
isAnotherDay(prevIndex: number, currIndex: number) {
if (currIndex == 0) return true;
return (
new Date(this.dispatcherHistory[prevIndex].timestampFrom).getDate() !=
new Date(this.dispatcherHistory[currIndex].timestampFrom).getDate()
);
} }
} }
}); });
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@import '../../../styles/animations.scss'; @use '../../../styles/journal-section';
@import '../../../styles/responsive.scss';
@import '../../../styles/badge.scss';
@import '../../../styles/variables.scss';
@import '../../../styles/JournalSection.scss';
table.dispatchers-table {
--_bg-table: #111;
--_bg-head: #101010;
--_bg-row: #2f2f2f;
width: 100%;
border-collapse: collapse;
position: relative;
text-align: center;
margin-bottom: 1em;
thead {
position: sticky;
top: 0;
background-color: var(--_bg-head);
}
th {
padding: 0.5em;
}
tr {
background-color: var(--_bg-row);
border-bottom: 2px solid black;
&:last-child {
border: none;
}
}
td {
padding: 0.75em;
.level-badge {
margin: 0 auto;
}
}
}
.text {
&--online {
color: springgreen;
}
&--offline {
color: #ddd;
}
}
</style> </style>
+7 -6
View File
@@ -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>
&nbsp;&bull;&nbsp; &nbsp;&bull;&nbsp;
<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>
+9 -10
View File
@@ -9,7 +9,7 @@
ref="button" ref="button"
> >
<img src="/images/icon-filter2.svg" alt="Open filters" /> <img src="/images/icon-filter2.svg" alt="Open filters" />
{{ $t('options.filters') }} [F] [F] {{ $t('options.filters') }}
<span class="active-indicator" v-if="currentOptionsActive"></span> <span class="active-indicator" v-if="currentOptionsActive"></span>
</button> </button>
@@ -33,7 +33,7 @@
<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">{{ <label v-if="propName == 'search-date-from'" for="search-date">{{
$t(`options.search-${optionsType}-date`) $t(`options.search-${optionsType}-date`)
}}</label> }}</label>
@@ -45,13 +45,13 @@
@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"
@@ -188,14 +188,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 +283,6 @@ export default defineComponent({
}, },
searchConfirm() { searchConfirm() {
this.$emit('onSearchConfirm');
this.handleRouteParams(); this.handleRouteParams();
}, },
@@ -301,6 +300,6 @@ 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';
</style> </style>
+11 -6
View File
@@ -30,7 +30,11 @@
</div> </div>
<transition name="dropdown-anim"> <transition name="dropdown-anim">
<div class="dropdown_wrapper" v-if="currentStatsTab !== null"> <div
class="dropdown_wrapper"
:class="{ 'dropdown-align-right': true }"
v-if="currentStatsTab !== null"
>
<keep-alive> <keep-alive>
<component :is="currentStatsTab" :key="currentStatsTab"></component> <component :is="currentStatsTab" :key="currentStatsTab"></component>
</keep-alive> </keep-alive>
@@ -75,11 +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_wrapper.dropdown-align-right {
max-width: 100%; left: auto;
right: 0;
max-width: 700px;
} }
</style> </style>
@@ -0,0 +1,311 @@
<template>
<div>
<div class="details-actions">
<button class="btn--action" @click="toggleExtraInfo">
<b>{{ $t('journal.entry-details') }}</b>
<img :src="`/images/icon-arrow-${showExtraInfo ? 'asc' : 'desc'}.svg`" alt="Arrow icon" />
</button>
<router-link
v-if="driverRouteLocation !== null"
class="a-button btn--action btn-timetable"
:to="driverRouteLocation"
>
<img src="/images/icon-train.svg" alt="train icon" />
<b>{{ $t('journal.timetable-online-button') }}</b>
</router-link>
</div>
<div class="details-body" v-if="showExtraInfo">
<div class="g-separator"></div>
<EntryStops :timetable="timetable" />
<div class="g-separator"></div>
<div class="timetable-specs">
<span class="badge specs-badge" v-if="timetable.authorName">
<span>{{ $t('journal.dispatcher-name') }}</span>
<span>{{ timetable.authorName }}</span>
</span>
<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>{{ timetable.maxSpeed }}km/h</span>
</span>
</div>
<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>
{{
currentHistoryIndex == 0
? timetable.stockLength
: stockHistory[currentHistoryIndex].stockLength || timetable.stockLength
}}m
</span>
</span>
<span class="badge specs-badge" v-if="timetable.stockMass">
<span>{{ $t('journal.stock-mass') }}</span>
<span>
{{
Math.floor(
(currentHistoryIndex == 0
? timetable.stockMass
: stockHistory[currentHistoryIndex].stockMass || timetable.stockMass) / 1000
)
}}t
</span>
</span>
</div>
<div class="stock-history">
<button class="btn btn--action" @click="copyStockToClipboard()">
<i class="fa-regular fa-copy"></i> {{ $t('journal.stock-copy') }}
</button>
<button
v-for="(sh, i) in stockHistory"
:key="i"
class="btn--action"
:data-checked="i == currentHistoryIndex"
@click.stop="currentHistoryIndex = i"
>
{{ sh.updatedAt }}
</button>
</div>
<div v-if="timetable.stockString" style="margin-top: 1em">
<StockList
:trainStockList="
(currentHistoryIndex == 0
? timetable.stockString
: stockHistory[currentHistoryIndex].stockString
).split(';')
"
/>
</div>
</div>
</div>
</div>
</template>
<script lang="ts">
import { PropType, defineComponent } from 'vue';
import StockList from '../../Global/StockList.vue';
import { API } from '../../../typings/api';
import { RouteLocationRaw } from 'vue-router';
import EntryStops from './EntryStops.vue';
import { useI18n } from 'vue-i18n';
export default defineComponent({
components: { StockList, EntryStops },
emits: ['toggleExtraInfo'],
props: {
showExtraInfo: {
type: Boolean,
required: true
},
timetable: {
type: Object as PropType<API.TimetableHistory.Data>,
required: true
}
},
data() {
return {
currentHistoryIndex: 0,
i18n: useI18n()
};
},
computed: {
stockHistory() {
return this.timetable.stockHistory
.slice()
.reverse()
.map((h) => {
const historyData = h.split('@');
return {
updatedAt: new Date(Number(historyData[0])).toLocaleTimeString(this.$i18n.locale, {
hour: '2-digit',
minute: '2-digit'
}),
stockString: historyData[1],
stockMass: Number(historyData[2]) || undefined,
stockLength: Number(historyData[3]) || undefined
};
});
},
driverRouteLocation(): RouteLocationRaw | null {
if (this.timetable.terminated) return null;
return {
name: 'DriverView',
query: {
trainId: `${this.timetable.driverId}|${this.timetable.trainNo}|eu`
}
};
}
},
methods: {
onImageError(e: Event) {
const imageEl = e.target as HTMLImageElement;
imageEl.src = '/images/icon-unknown.png';
},
toggleExtraInfo() {
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>
<style lang="scss" scoped>
@use '../../../styles/responsive';
@use '../../../styles/badge';
.details-body {
margin-top: 0.5em;
}
.details-actions {
display: flex;
gap: 0.5em;
margin-top: 1em;
button img {
height: 1.25em;
}
}
.stock-history {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
margin-top: 1em;
button[data-checked='true'] {
color: var(--clr-primary);
}
}
.timetable-specs,
.stock-specs {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
}
.specs-badge {
margin: 0;
span:first-child {
color: white;
background-color: #666;
border-radius: 0.25em 0 0 0.25em;
}
span:last-child {
color: black;
background-color: var(--clr-primary);
border-radius: 0 0.25em 0.25em 0;
}
}
hr {
margin: 0.5em 0;
}
.stock-dangers ul {
list-style: disc;
padding-left: 1em;
padding-top: 0.5em;
white-space: pre-wrap;
}
.dangers-notes {
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;
}
.details-actions {
justify-content: center;
}
}
</style>
@@ -1,20 +1,50 @@
<template> <template>
<div class="item-general"> <div class="item-general">
<span <span class="general-train">
class="general-train"
tabindex="0"
@click.stop="showTimetable(timetable, $event.currentTarget)"
@keydown.enter="showTimetable(timetable, $event.currentTarget)"
>
<span class="text--grayed">#{{ timetable.id }}</span> <span class="text--grayed">#{{ timetable.id }}</span>
<span class="badges" v-if="timetable.skr || timetable.twr"> <span
<span class="train-badge twr" v-if="timetable.twr" :title="$t('general.TWR')">TWR</span> class="train-badge twr"
<span class="train-badge skr" v-if="timetable.skr" :title="$t('general.SKR')">SKR</span> v-if="timetable.twr"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('warnings.TWR')"
>
TWR
</span>
<span
class="train-badge skr"
v-if="timetable.skr"
data-tooltip-type="BaseTooltip"
:data-tooltip-content="$t('warnings.SKR')"
>
SKR
</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> <span>
<strong class="text--primary"> <strong
data-tooltip-type="BaseTooltip"
:data-tooltip-content="getCategoryExplanation(timetable.trainCategoryCode)"
class="text--primary tooltip-help"
>
{{ timetable.trainCategoryCode }} {{ timetable.trainCategoryCode }}
</strong> </strong>
<strong>&nbsp;{{ timetable.trainNo }}</strong> <strong>&nbsp;{{ timetable.trainNo }}</strong>
@@ -28,17 +58,19 @@
{{ timetable.driverLevel < 2 ? 'L' : `${timetable.driverLevel}` }} {{ timetable.driverLevel < 2 ? 'L' : `${timetable.driverLevel}` }}
</strong> </strong>
<strong <router-link
v-if="isDonator(timetable.driverName)" v-if="apiStore.donatorsData.includes(timetable.driverName)"
class="text--donator" class="text--donator"
:title="$t('donations.driver-message')" data-tooltip-type="DonatorTooltip"
:data-tooltip-content="$t('donations.driver-message')"
:to="`/journal/timetables?search-driver=${timetable.driverName}`"
> >
{{ timetable.driverName }} <strong>{{ timetable.driverName }}</strong>
</strong> </router-link>
<strong v-else> <router-link v-else :to="`/journal/timetables?search-driver=${timetable.driverName}`">
{{ timetable.driverName }} <strong>{{ timetable.driverName }}</strong>
</strong> </router-link>
</span> </span>
<span class="general-time"> <span class="general-time">
@@ -62,8 +94,8 @@
!timetable.terminated !timetable.terminated
? $t('journal.timetable-active') ? $t('journal.timetable-active')
: timetable.fulfilled : timetable.fulfilled
? $t('journal.timetable-fulfilled') ? $t('journal.timetable-fulfilled')
: `${$t('journal.timetable-abandoned')} ${localeTime(timetable.endDate, $i18n.locale)}` : `${$t('journal.timetable-abandoned')} ${localeTime(timetable.endDate, $i18n.locale)}`
}} }}
</b> </b>
</span> </span>
@@ -75,33 +107,31 @@ import { PropType, defineComponent } from 'vue';
import { API } from '../../../typings/api'; import { API } from '../../../typings/api';
import dateMixin from '../../../mixins/dateMixin'; import dateMixin from '../../../mixins/dateMixin';
import modalTrainMixin from '../../../mixins/modalTrainMixin';
import styleMixin from '../../../mixins/styleMixin'; import styleMixin from '../../../mixins/styleMixin';
import donatorMixin from '../../../mixins/donatorMixin'; import { useApiStore } from '../../../store/apiStore';
import trainCategoryMixin from '../../../mixins/trainCategoryMixin';
export default defineComponent({ export default defineComponent({
mixins: [dateMixin, modalTrainMixin, styleMixin, donatorMixin], mixins: [dateMixin, styleMixin, trainCategoryMixin],
data() {
return {
apiStore: useApiStore()
};
},
props: { props: {
timetable: { timetable: {
type: Object as PropType<API.TimetableHistory.Data>, type: Object as PropType<API.TimetableHistory.Data>,
required: true required: true
} }
},
methods: {
showTimetable(timetable: API.TimetableHistory.Data, target: EventTarget | null) {
if (timetable?.terminated) return;
this.selectModalTrain(timetable.driverName + timetable.trainNo.toString(), target);
}
} }
}); });
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@import '../../../styles/responsive.scss'; @use '../../../styles/responsive';
@import '../../../styles/badge.scss'; @use '../../../styles/badge';
.item-general { .item-general {
display: flex; display: flex;
@@ -113,8 +143,21 @@ export default defineComponent({
margin-bottom: 0.5em; margin-bottom: 0.5em;
} }
.info-date { .general-train {
margin-right: 0.5em; display: flex;
justify-content: center;
align-items: center;
flex-wrap: wrap;
gap: 0.25em;
cursor: pointer;
}
.general-time {
display: flex;
align-items: center;
gap: 0.5em;
} }
.badges { .badges {
@@ -139,17 +182,18 @@ export default defineComponent({
} }
} }
.general-train { .btn-timetable {
cursor: pointer;
display: flex; display: flex;
flex-wrap: wrap; padding: 0.2em 0.5em;
justify-content: center;
align-items: center; img {
gap: 0.25em; height: 1.25em;
}
} }
@include smallScreen { @include responsive.smallScreen{
.item-general { .item-general {
flex-direction: column;
justify-content: center; justify-content: center;
} }
} }
@@ -1,5 +1,5 @@
<template> <template>
<div class="item-status" style="margin: 0.5em 0"> <div class="entry-status" style="margin: 0.5em 0">
<ProgressBar <ProgressBar
:progressPercent="~~((timetable.currentDistance / timetable.routeDistance) * 100)" :progressPercent="~~((timetable.currentDistance / timetable.routeDistance) * 100)"
:progressType="!timetable.fulfilled && timetable.terminated ? 'abandoned' : ''" :progressType="!timetable.fulfilled && timetable.terminated ? 'abandoned' : ''"
@@ -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,16 +59,21 @@ export default defineComponent({
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@import '../../../styles/responsive.scss'; @use '../../../styles/responsive';
.item-status { .entry-status {
display: flex; display: flex;
align-items: center; align-items: center;
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>
@@ -0,0 +1,303 @@
<template>
<div class="entry-stops">
<ul class="stop-list">
<li v-for="(stop, i) in timetableStops" :key="stop.stopName">
<span class="stop-label" :data-confirmed="stop.isConfirmed">
<span v-if="i > 0">&gt;</span>
<span class="stop-name">{{ stop.stopName }}</span>
<span
class="stop-date"
v-if="stop.scheduledArrivalTimestamp != 0"
:data-delayed="
stop.isConfirmed && stop.arrivalTimestamp - stop.scheduledArrivalTimestamp > 0
"
:data-preponed="
stop.isConfirmed &&
stop.arrivalTimestamp != 0 &&
stop.arrivalTimestamp - stop.scheduledArrivalTimestamp < 0
"
>
<span
v-if="stop.isConfirmed && stop.arrivalTimestamp - stop.scheduledArrivalTimestamp != 0"
>
p. <s>{{ timestampToString(stop.scheduledArrivalTimestamp) }}</s>
{{ timestampToString(stop.arrivalTimestamp) }}
</span>
<span v-else>p. {{ timestampToString(stop.scheduledArrivalTimestamp) }}</span>
</span>
<span
class="stop-time"
v-if="stop.stopTime > 0"
:data-stop-ph="stop.stopType.includes('ph')"
:data-stop-pt="stop.stopType.includes('pt')"
:data-stop-pm="stop.stopType.includes('pm')"
>
/<span>{{ stop.stopTime }} {{ stop.stopType }}</span
>/
</span>
<span
class="stop-date"
v-if="
stop.scheduledDepartureTimestamp != 0 &&
stop.scheduledArrivalTimestamp != stop.scheduledDepartureTimestamp
"
:data-delayed="
stop.isConfirmed && stop.departureTimestamp - stop.scheduledDepartureTimestamp > 0
"
:data-preponed="
stop.isConfirmed &&
stop.departureTimestamp != 0 &&
stop.departureTimestamp - stop.scheduledDepartureTimestamp < 0
"
>
<span
v-if="
stop.isConfirmed && stop.departureTimestamp - stop.scheduledDepartureTimestamp != 0
"
>
o. <s>{{ timestampToString(stop.scheduledDepartureTimestamp) }}</s>
{{ timestampToString(stop.departureTimestamp) }}
</span>
<span v-else>o. {{ timestampToString(stop.scheduledDepartureTimestamp) }}</span>
</span>
</span>
</li>
</ul>
<ul class="timetable-path-list" v-if="timetablePathDetails">
<li
v-for="(pathData, i) in timetablePathDetails"
:data-visited="pathData.isVisited"
:data-next-visited="
i < timetablePathDetails.length - 1 && timetablePathDetails[i + 1].isVisited
"
>
<span v-if="i > 0" class="path-arrow">&gt;</span>
<span class="path-arrival" v-if="pathData.arrival">{{ pathData.arrival }}</span>
<b class="path-scenery">{{ pathData.sceneryName }}</b>
<span class="path-departure" v-if="pathData.departure">{{ pathData.departure }}</span>
</li>
</ul>
</div>
</template>
<script lang="ts">
import { PropType, defineComponent } from 'vue';
import dateMixin from '../../../mixins/dateMixin';
import { API } from '../../../typings/api';
interface ITimetableStopDetails {
stopName: string;
stopComments: string | null;
stopTime: number;
stopType: string;
arrivalTimestamp: number;
scheduledArrivalTimestamp: number;
departureTimestamp: number;
scheduledDepartureTimestamp: number;
isConfirmed: boolean;
}
export default defineComponent({
mixins: [dateMixin],
props: {
timetable: {
type: Object as PropType<API.TimetableHistory.Data>,
required: true
}
},
computed: {
timetablePathDetails() {
if (!this.timetable.path || this.timetable.path == '') return null;
return this.timetable.path.split(';').map((pathEl, i) => {
const [arrival, name, departure] = pathEl.split(',');
const sceneryName = name.split(' ').slice(0, -1).join(' ');
const sceneryHash = name.split(' ').pop()?.replace('.sc', '') ?? '';
const isVisited = this.timetable.visitedSceneries.includes(sceneryHash);
return {
arrival,
sceneryName,
sceneryHash,
departure,
isVisited,
isVisitedOffline:
!isVisited &&
this.timetable.visitedSceneries.includes(`${sceneryName} ${sceneryHash}.sc`)
};
});
},
timetableStops(): ITimetableStopDetails[] {
const timetable = this.timetable;
const stopNames = timetable.sceneriesString.split('%');
return stopNames.reduce<ITimetableStopDetails[]>((acc, stopName, i, arr) => {
const arrivalDate =
i == arr.length - 1
? (timetable.checkpointArrivals.at(i) ?? timetable.endDate)
: timetable.checkpointArrivals.at(i);
const scheduledArrivalDate =
i == arr.length - 1
? (timetable.checkpointArrivalsScheduled.at(i) ?? timetable.scheduledEndDate)
: timetable.checkpointArrivalsScheduled.at(i);
const departureDate =
i == 0
? (timetable.checkpointDepartures.at(i) ?? timetable.beginDate)
: timetable.checkpointDepartures.at(i);
const scheduledDepartureDate =
i == 0
? (timetable.checkpointDeparturesScheduled.at(i) ?? timetable.scheduledBeginDate)
: timetable.checkpointDeparturesScheduled.at(i);
const stopTime = Number(timetable.checkpointStopTypes.at(i)?.split(',')[0]) || 0;
const stopType = timetable.checkpointStopTypes.at(i)?.split(',').slice(1).join(',') || 'pt';
const stopComments = timetable.checkpointComments.at(i) ?? null;
acc.push({
stopName,
stopTime,
stopType,
stopComments,
arrivalTimestamp: this.dateStringToTimestamp(arrivalDate),
scheduledArrivalTimestamp: this.dateStringToTimestamp(scheduledArrivalDate),
departureTimestamp: this.dateStringToTimestamp(departureDate),
scheduledDepartureTimestamp: this.dateStringToTimestamp(scheduledDepartureDate),
isConfirmed: i < timetable.confirmedStopsCount
});
return acc;
}, []);
}
}
});
</script>
<style lang="scss" scoped>
@use '../../../styles/badge';
.entry-stops {
word-wrap: break-word;
gap: 0.25em;
font-size: 0.95em;
}
.stop-list {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
padding: 0.5em 0;
}
.stop-label {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
align-items: center;
color: white;
&[data-confirmed='true'] > .stop-name {
color: lightgreen;
}
&[data-confirmed='true'] > .stop-date:not([data-preponed='true']):not([data-delayed='true']) {
color: lightgreen;
}
}
.stop-name {
font-weight: bold;
color: #ccc;
i {
display: none;
}
}
.stop-date {
color: #ccc;
s {
color: #aaa;
}
&[data-delayed='true'] {
color: salmon;
}
&[data-preponed='true'] {
color: mediumspringgreen;
}
}
.stop-time {
&[data-stop-pt='true'] span {
color: #999;
}
&[data-stop-ph='true'] span,
&[data-stop-pm='true'] span {
color: gold;
}
}
.timetable-path-list {
display: flex;
flex-wrap: wrap;
gap: 0.5em 0;
padding: 0.5em 0;
color: #ccc;
li > .path-scenery:first-child,
li > .path-arrival:nth-child(2) {
border-radius: 0.5em 0 0 0.5em;
}
li > :last-child {
border-radius: 0 0.5em 0.5em 0;
}
}
.path-scenery {
padding: 0.25em 0.5em;
background-color: #303030;
}
.path-arrival,
.path-departure {
padding: 0.25em;
display: inline-block;
background-color: #4e4e4e;
min-width: 25px;
text-align: center;
}
.path-arrow {
padding: 0 0.5em;
}
.timetable-path-list > li[data-visited='true'] {
.path-arrival,
.path-scenery,
.path-arrow {
color: lightgreen;
}
&[data-next-visited='true'] .path-departure {
color: lightgreen;
}
}
</style>
@@ -12,37 +12,70 @@
<hr class="header-separator" /> <hr class="header-separator" />
<div class="info-stats"> <div class="info-stats">
<span class="stat-badge"> <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="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>
</span> </span>
<span class="stat-badge"> <span class="badge stat-badge">
<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>
<span class="stat-badge"> <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>{{ $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>
<span class="stat-badge"> <span class="badge stat-badge">
<span>{{ $t('journal.driver-stats.stations') }}</span> <span>{{ $t('journal.driver-stats.stations') }}</span>
<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>
@@ -0,0 +1,154 @@
<template>
<li class="timetable-history-entry">
<!-- General -->
<EntryGeneral :timetable="timetableEntry" />
<!-- Route -->
<div class="entry-route">
<b>{{ timetableEntry.route.replace('|', ' - ') }}</b>
</div>
<hr />
<div @click="toggleExtraInfo" style="cursor: pointer">
<!-- Status -->
<EntryStatus :timetable="timetableEntry" />
</div>
<!-- Extra -->
<EntryDetails
:timetable="timetableEntry"
:show-extra-info="showExtraInfo"
@toggle-extra-info="toggleExtraInfo"
/>
</li>
</template>
<script lang="ts">
import { defineComponent, PropType } from 'vue';
import { API } from '../../../typings/api';
import { useApiStore } from '../../../store/apiStore';
import { Journal } from '../typings';
import trainCategoryMixin from '../../../mixins/trainCategoryMixin';
import dateMixin from '../../../mixins/dateMixin';
import styleMixin from '../../../mixins/styleMixin';
import EntryGeneral from './EntryGeneral.vue';
import EntryStatus from './EntryStatus.vue';
import EntryDetails from './EntryDetails.vue';
export default defineComponent({
props: {
timetableEntry: {
type: Object as PropType<API.TimetableHistory.Data>,
required: true
},
showExtraInfo: {
type: Boolean,
required: true
}
},
components: { EntryDetails, EntryGeneral, EntryStatus },
mixins: [trainCategoryMixin, dateMixin, styleMixin],
emits: ['toggleShowExtraInfo'],
data() {
return {
apiStore: useApiStore()
};
},
computed: {
timetablePathDetails() {
if (!this.timetableEntry.path || this.timetableEntry.path == '') return null;
return this.timetableEntry.path.split(';').map((pathEl, i) => {
const [arrival, name, departure] = pathEl.split(',');
const sceneryName = name.split(' ').slice(0, -1).join(' ');
const sceneryHash = name.split(' ').pop()?.replace('.sc', '') ?? '';
return {
arrival,
sceneryName,
sceneryHash,
departure,
isVisited: this.timetableEntry.visitedSceneries?.includes(sceneryHash) ?? false
};
});
},
timetableStops(): Journal.TimetableStopDetails[] {
const timetableEntry = this.timetableEntry;
const stopNames = timetableEntry.sceneriesString.split('%');
return stopNames.reduce<Journal.TimetableStopDetails[]>((acc, stopName, i, arr) => {
const arrivalDate =
i == arr.length - 1
? (timetableEntry.checkpointArrivals.at(i) ?? timetableEntry.endDate)
: timetableEntry.checkpointArrivals.at(i);
const scheduledArrivalDate =
i == arr.length - 1
? (timetableEntry.checkpointArrivalsScheduled.at(i) ?? timetableEntry.scheduledEndDate)
: timetableEntry.checkpointArrivalsScheduled.at(i);
const departureDate =
i == 0
? (timetableEntry.checkpointDepartures.at(i) ?? timetableEntry.beginDate)
: timetableEntry.checkpointDepartures.at(i);
const scheduledDepartureDate =
i == 0
? (timetableEntry.checkpointDeparturesScheduled.at(i) ??
timetableEntry.scheduledBeginDate)
: timetableEntry.checkpointDeparturesScheduled.at(i);
const stopTime = Number(timetableEntry.checkpointStopTypes.at(i)?.split(',')[0]) || 0;
const stopType = timetableEntry.checkpointStopTypes.at(i)?.split(',')[1] || '';
acc.push({
stopName,
arrivalTimestamp: this.dateStringToTimestamp(arrivalDate),
scheduledArrivalTimestamp: this.dateStringToTimestamp(scheduledArrivalDate),
departureTimestamp: this.dateStringToTimestamp(departureDate),
scheduledDepartureTimestamp: this.dateStringToTimestamp(scheduledDepartureDate),
stopTime,
stopType,
isConfirmed: i < timetableEntry.confirmedStopsCount
});
return acc;
}, []);
}
},
methods: {
toggleExtraInfo() {
this.$emit('toggleShowExtraInfo');
}
}
});
</script>
<style lang="scss" scoped>
@use '../../../styles/responsive';
.timetable-history-entry {
background-color: #1a1a1a;
padding: 1em;
}
.entry-route {
display: flex;
}
@include responsive.smallScreen{
.entry-route {
justify-content: center;
text-align: center;
}
}
</style>
@@ -1,35 +1,40 @@
<template> <template>
<div> <div>
<transition name="status-anim" mode="out-in"> <div class="journal_warning" v-if="store.isOffline">
<div :key="dataStatus"> {{ $t('app.offline') }}
<div class="journal_warning" v-if="store.isOffline"> </div>
{{ $t('app.offline') }}
</div>
<Loading v-else-if="dataStatus == Status.Data.Loading" /> <Loading v-else-if="dataStatus == Status.Data.Loading" />
<div v-else-if="dataStatus == Status.Data.Error" class="journal_warning error"> <div v-else-if="dataStatus == Status.Data.Error" class="journal_warning error">
{{ $t('app.error') }} {{ $t('app.error') }}
</div> </div>
<div v-else-if="timetableHistory.length == 0" class="journal_warning"> <div v-else-if="timetableHistory.length == 0" class="journal_warning">
{{ $t('app.no-result') }} {{ $t('app.no-result') }}
</div> </div>
<div v-else> <div v-else>
<TimetableHistoryList :timetableHistory="timetableHistory" /> <transition-group name="list-anim" class="journal-list" tag="ul">
<JournalTimetableEntry
v-for="(timetableEntry, i) in timetableHistory"
:key="timetableEntry.id"
:timetableEntry="timetableEntry"
:onToggleShowExtraInfo="() => toggleExtraInfo(timetableEntry.id)"
:showExtraInfo="extraInfoIndexes.includes(timetableEntry.id)"
/>
</transition-group>
<AddDataButton <AddDataButton
:list="timetableHistory" :list="timetableHistory"
:scrollDataLoaded="scrollDataLoaded" :scrollDataLoaded="scrollDataLoaded"
:scrollNoMoreData="scrollNoMoreData" :scrollNoMoreData="scrollNoMoreData"
@addHistoryData="addHistoryData" @addHistoryData="addHistoryData"
/> />
</div> </div>
</div>
</transition>
<div class="journal_warning" v-if="scrollNoMoreData">{{ $t('journal.no-further-data') }}</div> <div class="journal_warning" v-if="scrollNoMoreData">{{ $t('journal.no-further-data') }}</div>
<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>
@@ -41,13 +46,18 @@ import { defineComponent, PropType } from 'vue';
import Loading from '../../Global/Loading.vue'; import Loading from '../../Global/Loading.vue';
import AddDataButton from '../../Global/AddDataButton.vue'; import AddDataButton from '../../Global/AddDataButton.vue';
import TimetableHistoryList from './TimetableHistoryList.vue'; import JournalTimetableEntry from './JournalTimetableEntry.vue';
import { useMainStore } from '../../../store/mainStore'; import { useMainStore } from '../../../store/mainStore';
import { Status } from '../../../typings/common'; import { Status } from '../../../typings/common';
import { API } from '../../../typings/api'; import { API } from '../../../typings/api';
export default defineComponent({ export default defineComponent({
components: { Loading, AddDataButton, TimetableHistoryList }, components: {
Loading,
AddDataButton,
JournalTimetableEntry
},
props: { props: {
timetableHistory: { timetableHistory: {
@@ -71,13 +81,44 @@ export default defineComponent({
data() { data() {
return { return {
Status, Status,
store: useMainStore() store: useMainStore(),
extraInfoIndexes: [] as number[]
}; };
},
watch: {
'$route.query': {
deep: true,
handler() {
this.extraInfoIndexes.length = 0;
}
}
},
methods: {
toggleExtraInfo(id: number) {
const existingIdx = this.extraInfoIndexes.indexOf(id);
if (existingIdx != -1) this.extraInfoIndexes.splice(existingIdx, 1);
else this.extraInfoIndexes.push(id);
}
} }
}); });
</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 responsive.smallScreen{
.journal_item-info {
text-align: center;
}
.item-route {
display: flex;
justify-content: center;
}
}
</style> </style>
@@ -1,173 +0,0 @@
<template>
<div class="item-extra" v-if="timetable.stockString && timetable.stockMass && showExtraInfo">
<hr />
<div class="stock-specs">
<span class="badge">
<span>{{ $t('journal.dispatcher-name') }}</span>
<span>{{ timetable.authorName }}</span>
</span>
</div>
<div class="stock-specs">
<span class="badge">
<span>{{ $t('journal.stock-max-speed') }}</span>
<span>{{ timetable.maxSpeed }}km/h</span>
</span>
<span class="badge">
<span>{{ $t('journal.stock-length') }}</span>
<span>
{{
currentHistoryIndex == 0
? timetable.stockLength
: stockHistory[currentHistoryIndex].stockLength || timetable.stockLength
}}m
</span>
</span>
<span class="badge">
<span>{{ $t('journal.stock-mass') }}</span>
<span>
{{
Math.floor(
(currentHistoryIndex == 0
? timetable.stockMass!
: stockHistory[currentHistoryIndex].stockMass || timetable.stockMass) / 1000
)
}}t
</span>
</span>
</div>
<!-- Historia zmian w składzie -->
<div class="stock-history" v-if="stockHistory.length > 1">
<button
v-for="(sh, i) in stockHistory"
:key="i"
class="btn--action"
:data-checked="i == currentHistoryIndex"
@click.stop="currentHistoryIndex = i"
>
{{ sh.updatedAt }}
</button>
</div>
<StockList
:trainStockList="
(currentHistoryIndex == 0
? timetable.stockString
: stockHistory[currentHistoryIndex].stockString
).split(';')
"
/>
</div>
</template>
<script lang="ts">
import { PropType, defineComponent } from 'vue';
import StockList from '../../Global/StockList.vue';
import { API } from '../../../typings/api';
export default defineComponent({
components: { StockList },
props: {
showExtraInfo: {
type: Boolean,
required: true
},
timetable: {
type: Object as PropType<API.TimetableHistory.Data>,
required: true
}
},
data() {
return {
currentHistoryIndex: 0
};
},
computed: {
stockHistory() {
return this.timetable.stockHistory
.slice()
.reverse()
.map((h) => {
const historyData = h.split('@');
return {
updatedAt: new Date(Number(historyData[0])).toLocaleTimeString(this.$i18n.locale, {
hour: '2-digit',
minute: '2-digit'
}),
stockString: historyData[1],
stockMass: Number(historyData[2]) || undefined,
stockLength: Number(historyData[3]) || undefined
};
});
}
},
methods: {
onImageError(e: Event) {
const imageEl = e.target as HTMLImageElement;
imageEl.src = '/images/icon-unknown.png';
}
}
});
</script>
<style lang="scss" scoped>
@import '../../../styles/variables.scss';
@import '../../../styles/responsive.scss';
@import '../../../styles/badge.scss';
.item-extra {
margin-top: 0.5em;
}
.stock-history {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
margin-top: 1em;
button[data-checked='true'] {
color: $accentCol;
}
}
.stock-specs {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
margin-top: 0.5em;
.badge {
margin: 0;
span:last-child {
color: black;
background-color: $accentCol;
}
}
@include smallScreen() {
justify-content: center;
}
}
ul.stock-list {
display: flex;
align-items: flex-end;
overflow: auto;
padding-bottom: 0.5em;
li > div {
margin: 1em 0;
text-align: center;
color: #aaa;
font-size: 0.9em;
}
}
</style>
@@ -1,101 +0,0 @@
<template>
<ul class="journal-list">
<transition-group name="list-anim">
<li
v-for="{ timetable, showExtraInfo } in computedTimetableHistory"
class="journal_item"
:key="timetable.id"
@click="showExtraInfo.value = !showExtraInfo.value"
>
<div class="journal_item-info">
<!-- General -->
<TimetableGeneral :timetable="timetable" />
<!-- Route -->
<span class="item-route">
<b>{{ timetable.route.replace('|', ' - ') }}</b>
</span>
<hr />
<!-- Stops -->
<TimetableStops :timetable="timetable" :showExtraInfo="showExtraInfo.value" />
<!-- Status -->
<TimetableStatus :timetable="timetable" />
<button class="btn--option btn--show">
{{ $t('journal.stock-info') }}
<img
:src="`/images/icon-arrow-${showExtraInfo.value ? 'asc' : 'desc'}.svg`"
alt="Arrow icon"
/>
</button>
<!-- Extra -->
<TimetableExtra :timetable="timetable" :showExtraInfo="showExtraInfo.value" />
</div>
</li>
</transition-group>
</ul>
</template>
<script lang="ts">
import { PropType, defineComponent, ref } from 'vue';
import TimetableGeneral from './TimetableGeneral.vue';
import TimetableStops from './TimetableStops.vue';
import TimetableStatus from './TimetableStatus.vue';
import TimetableExtra from './TimetableExtra.vue';
import { API } from '../../../typings/api';
export default defineComponent({
components: { TimetableGeneral, TimetableStops, TimetableStatus, TimetableExtra },
props: {
timetableHistory: {
type: Array as PropType<API.TimetableHistory.Response>,
required: true
}
},
computed: {
computedTimetableHistory() {
return this.timetableHistory.map((timetable) => ({
timetable,
showExtraInfo: ref(false)
}));
}
}
});
</script>
<style lang="scss" scoped>
@import '../../../styles/variables.scss';
@import '../../../styles/responsive.scss';
@import '../../../styles/JournalSection.scss';
.btn--show {
display: flex;
font-weight: bold;
padding: 0.2em 0.45em;
img {
height: 1.3em;
}
}
hr {
margin: 0.25em 0;
}
@include smallScreen {
.journal_item-info {
text-align: center;
}
.item-route {
display: flex;
justify-content: center;
}
.btn--show {
margin: 1em auto 0 auto;
}
}
</style>
@@ -1,112 +0,0 @@
<template>
<div class="stop-list" v-if="showExtraInfo == true">
<span
v-for="(stop, i) in timetableStops.filter((_, i) =>
!showExtraInfo ? i == 0 || i == timetableStops.length - 1 : true
)"
class="stop-list-item"
:key="stop.stopName"
:data-confirmed="stop.confirmed"
>
<span v-if="i > 0">
&gt;
<span v-if="!showExtraInfo && i == 1 && timetableStops.length > 2">
... (+{{ timetableStops.length - 2 }}) &gt;
</span>
</span>
<span class="stop-name">{{ stop.stopName }}</span>
<span v-html="stop.html"></span>
</span>
</div>
</template>
<script lang="ts">
import { PropType, defineComponent } from 'vue';
import dateMixin from '../../../mixins/dateMixin';
import { API } from '../../../typings/api';
export default defineComponent({
mixins: [dateMixin],
props: {
showExtraInfo: {
type: Boolean,
required: true
},
timetable: {
type: Object as PropType<API.TimetableHistory.Data>,
required: true
}
},
computed: {
timetableStops() {
const timetable = this.timetable;
const stopNames = timetable.sceneriesString.split('%');
const beginDateHTML = ` (o. ${
timetable.beginDate != timetable.scheduledBeginDate
? `<s class="text--grayed">${this.localeTime(timetable.beginDate, this.$i18n.locale)}</s>`
: ''
} <span>${this.localeTime(timetable.scheduledBeginDate, this.$i18n.locale)}</span>)`;
const endDateHTML = ` (p. ${
timetable.endDate != timetable.scheduledEndDate && timetable.fulfilled
? `<s class="text--grayed">${this.localeTime(timetable.endDate, this.$i18n.locale)}</s>`
: ''
} <span>${this.localeTime(timetable.scheduledEndDate, this.$i18n.locale)}</span>)`;
return stopNames.map((stopName, i) => {
const confirmed = i < timetable.confirmedStopsCount;
if (i == 0) return { stopName, html: beginDateHTML, confirmed };
if (i == stopNames.length - 1) return { stopName, html: endDateHTML, confirmed };
const departureDateScheduled = this.stringToDate(
timetable.checkpointDeparturesScheduled?.at(i)
);
const departureDateReal = this.stringToDate(timetable.checkpointDepartures?.at(i));
const arrivalDateScheduled = this.stringToDate(
timetable.checkpointArrivalsScheduled?.at(i)
);
const arrivalDateReal = this.stringToDate(timetable.checkpointArrivals?.at(i));
const arrivalHTML =
(arrivalDateReal &&
arrivalDateScheduled &&
arrivalDateReal?.getTime() != arrivalDateScheduled?.getTime()
? `<s class="text--grayed">${this.parseDateToTimeString(arrivalDateScheduled)}</s> `
: '') + this.parseDateToTimeString(arrivalDateReal || arrivalDateScheduled);
const departureHTML =
(departureDateReal &&
departureDateScheduled &&
departureDateReal?.getTime() != departureDateScheduled?.getTime()
? `<s class="text--grayed">${this.parseDateToTimeString(departureDateScheduled)}</s> `
: '') + this.parseDateToTimeString(departureDateReal || departureDateScheduled);
let html = `${arrivalHTML}${departureHTML ? ` / ${departureHTML}` : ''}`;
if (html) html = ` (${html})`;
return { stopName, html, confirmed };
});
}
}
});
</script>
<style lang="scss" scoped>
.stop-list {
word-wrap: break-word;
gap: 0.25em;
font-size: 0.95em;
color: #adadad;
&-item[data-confirmed='true'] {
color: lightgreen;
.stop-name {
font-weight: bold;
}
}
}
</style>
+23 -4
View File
@@ -1,12 +1,18 @@
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-via';
export type TimetableSearchType = { export type TimetableSearchType = {
[key in TimetableSearchKey]: string; [key in TimetableSearchKey]: string;
@@ -17,7 +23,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;
@@ -37,6 +43,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'
} }
@@ -64,4 +72,15 @@ export namespace Journal {
iconName: string; iconName: string;
disabled: boolean; disabled: boolean;
} }
export interface TimetableStopDetails {
stopName: string;
arrivalTimestamp: number;
scheduledArrivalTimestamp: number;
departureTimestamp: number;
scheduledDepartureTimestamp: number;
stopTime: number;
stopType: string;
isConfirmed: boolean;
}
} }
@@ -1,31 +1,18 @@
<template> <template>
<section class="scenery-table-section"> <div class="scenery-dispatchers-history">
<Loading v-if="dataStatus != DataStatus.Loaded && historyList.length == 0" /> <div class="history-wrapper">
<Loading v-if="dataStatus != DataStatus.Loaded && historyList.length == 0" />
<div class="no-history" v-else-if="historyList.length == 0"> <div v-else-if="historyList.length == 0" class="no-history">
{{ $t('scenery.history-list-empty') }} {{ $t('scenery.history-list-empty') }}
</div> </div>
<table class="scenery-history-table" v-else> <div v-else class="journal-list">
<thead> <div v-for="historyItem in historyList" :key="historyItem.id">
<th>{{ $t('scenery.dispatchers-history-hash') }}</th> <span>
<th>{{ $t('scenery.dispatchers-history-dispatcher') }}</th> <span class="text--grayed" style="margin-right: 10px">
<th>{{ $t('scenery.dispatchers-history-level') }}</th> #{{ historyItem.stationHash }}
<th>{{ $t('scenery.dispatchers-history-rate') }}</th> </span>
<th>{{ $t('scenery.dispatchers-history-date') }}</th>
</thead>
<tbody>
<tr v-for="historyItem in historyList" :key="historyItem.id">
<td>#{{ historyItem.stationHash }}</td>
<td>
<router-link
:to="`/journal/dispatchers?search-dispatcher=${historyItem.dispatcherName}`"
>
<b>{{ historyItem.dispatcherName }}</b>
</router-link>
</td>
<td>
<b <b
v-if="historyItem.dispatcherLevel !== null" v-if="historyItem.dispatcherLevel !== null"
class="level-badge dispatcher" class="level-badge dispatcher"
@@ -35,55 +22,67 @@
> >
{{ historyItem.dispatcherLevel >= 2 ? historyItem.dispatcherLevel : 'L' }} {{ historyItem.dispatcherLevel >= 2 ? historyItem.dispatcherLevel : 'L' }}
</b> </b>
<b style="margin-left: 5px">
<router-link
:to="`/journal/dispatchers?search-dispatcher=${historyItem.dispatcherName}`"
>
{{ historyItem.dispatcherName }}
</router-link>
</b>
<b v-else>?</b> <div>
</td> <span>
<td class="text--primary"> {{ $t('scenery.dispatcher-rate') }}
<b>{{ historyItem.dispatcherRate }}</b> <b class="text--primary"> {{ historyItem.dispatcherRate }}</b>
</td> </span>
<td style="min-width: 300px"> |
<div v-if="historyItem.timestampTo"> <span>
{{ $t('scenery.dispatcher-status-changes') }}
<b class="text--primary">{{ historyItem.statusHistory.length }}</b>
</span>
</div>
</span>
<span>
<span v-if="historyItem.timestampTo">
<b>{{ $d(historyItem.timestampFrom) }}</b> <b>{{ $d(historyItem.timestampFrom) }}</b>
{{ timestampToString(historyItem.timestampFrom) }} {{ timestampToString(historyItem.timestampFrom) }}
- {{ timestampToString(historyItem.timestampTo) }} ({{ - {{ timestampToString(historyItem.timestampTo) }} ({{
calculateDuration(historyItem.currentDuration) calculateDuration(historyItem.currentDuration)
}}) }})
</div> </span>
<div class="dispatcher-online" v-else> <span class="dispatcher-online" v-else>
{{ $t('journal.online-since') }} {{ $t('journal.online-since') }}
<b>{{ timestampToString(historyItem.timestampFrom) }}</b> <b>{{ timestampToString(historyItem.timestampFrom) }}</b>
({{ calculateDuration(historyItem.currentDuration) }}) ({{ calculateDuration(historyItem.currentDuration) }})
</div> </span>
</td> </span>
</tr> </div>
</tbody> </div>
</table> </div>
</section>
<div class="bottom-info"> <div class="bottom-info">
<button class="btn btn--option" v-if="historyList.length > 0" @click="navigateToHistory"> <button class="btn btn--option" v-if="historyList.length > 0" @click="navigateToHistory">
{{ $t('scenery.bottom-info') }} {{ $t('scenery.bottom-info') }}
</button> </button>
</div>
</div> </div>
</template> </template>
<script lang="ts"> <script lang="ts">
import { defineComponent, PropType } from 'vue'; import { defineComponent, PropType } from 'vue';
import dateMixin from '../../mixins/dateMixin'; import dateMixin from '../../mixins/dateMixin';
import Station from '../../scripts/interfaces/Station';
import Loading from '../Global/Loading.vue'; import Loading from '../Global/Loading.vue';
import styleMixin from '../../mixins/styleMixin'; import styleMixin from '../../mixins/styleMixin';
import listObserverMixin from '../../mixins/listObserverMixin';
import { ActiveScenery } from '../../store/typings';
import { API } from '../../typings/api'; import { API } from '../../typings/api';
import { Status } from '../../typings/common'; import { ActiveScenery, Station, Status } from '../../typings/common';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
export default defineComponent({ export default defineComponent({
name: 'SceneryDispatchersHistory', name: 'SceneryDispatchersHistory',
mixins: [dateMixin, styleMixin, listObserverMixin], mixins: [dateMixin, styleMixin],
components: { Loading }, components: { Loading },
props: { props: {
station: { station: {
@@ -149,25 +148,57 @@ 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 {
height: 100%;
overflow: auto;
display: grid;
gap: 0.5em;
grid-template-rows: auto 40px;
}
.history-wrapper {
position: relative;
overflow: auto;
}
.journal-list {
display: flex;
flex-direction: column;
gap: 0.5em;
text-align: left;
}
.journal-list > div {
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: wrap;
gap: 0.5em;
padding: 0.5em;
background-color: #2b2b2b;
line-height: 1.75em;
}
.level-badge { .level-badge {
margin: 0 auto; text-align: center;
display: inline-block;
line-height: 1.6em;
} }
.dispatcher-online { .dispatcher-online {
color: springgreen; color: springgreen;
} }
@include smallScreen { @include responsive.smallScreen{
.history-list { .journal-list > div {
font-size: 1.1em;
}
.list-item {
align-items: center;
flex-direction: column; flex-direction: column;
justify-content: center;
text-align: center;
} }
} }
</style> </style>
../../store/storeTypes
+2 -4
View File
@@ -14,8 +14,7 @@
<script lang="ts"> <script lang="ts">
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import Station from '../../scripts/interfaces/Station'; import { ActiveScenery, Station } from '../../typings/common';
import { ActiveScenery } from '../../store/typings';
export default defineComponent({ export default defineComponent({
props: { props: {
@@ -36,8 +35,7 @@ export default defineComponent({
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@import '../../styles/variables.scss'; @use '../../styles/responsive';
@import '../../styles/responsive.scss';
.info-header { .info-header {
margin-top: 1em; margin-top: 1em;
+14 -87
View File
@@ -1,69 +1,11 @@
<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" />
<SceneryInfoRoutes v-if="station" :station="station" />
<SceneryInfoAuthors :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>
&bull; <b>{{ $t('controls.title') }}:</b>
{{ $t(`controls.${station.generalInfo.controlType}`) }}
</span>
<span>
&bull; <b>{{ $t('signals.title') }}:</b>
{{ $t(`signals.${station.generalInfo.signalType}`) }}
</span>
<span v-if="station.generalInfo.lines">
&bull; <b>{{ $t('scenery.lines-title') }}:</b> {{ station.generalInfo.lines }}
</span>
<span v-if="station.generalInfo.project">
&bull; <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" />
<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> <div style="margin: 2em 0; height: 2px; background-color: white"></div>
@@ -72,7 +14,7 @@
<div class="info-lists"> <div class="info-lists">
<!-- user list --> <!-- user list -->
<SceneryInfoUserList :onlineScenery="onlineScenery" /> <SceneryInfoUserList :onlineScenery="onlineScenery" :station="station" />
<!-- spawn list --> <!-- spawn list -->
<SceneryInfoSpawnList :onlineScenery="onlineScenery" /> <SceneryInfoSpawnList :onlineScenery="onlineScenery" />
@@ -89,16 +31,20 @@ 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 Station from '../../scripts/interfaces/Station'; import SceneryInfoGeneral from './SceneryInfo/SceneryInfoGeneral.vue';
import { ActiveScenery } from '../../store/typings'; import SceneryInfoAuthors from "./SceneryInfo/SceneryInfoAuthors.vue";
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,
}, },
props: { props: {
station: { station: {
@@ -113,8 +59,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;
@@ -125,11 +71,6 @@ h3.section-header {
align-items: center; align-items: center;
font-size: 1.2em; font-size: 1.2em;
img {
width: 1.1em;
margin-left: 0.5em;
}
} }
.info-lists { .info-lists {
@@ -140,20 +81,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>
@@ -14,7 +14,7 @@
> >
<span <span
class="text--donator" class="text--donator"
v-if="isDonator(onlineScenery.dispatcherName)" v-if="apiStore.donatorsData.includes(onlineScenery.dispatcherName)"
:title="$t('donations.dispatcher-message')" :title="$t('donations.dispatcher-message')"
> >
{{ onlineScenery.dispatcherName }} {{ onlineScenery.dispatcherName }}
@@ -49,11 +49,18 @@ import dateMixin from '../../../mixins/dateMixin';
import routerMixin from '../../../mixins/routerMixin'; import routerMixin from '../../../mixins/routerMixin';
import styleMixin from '../../../mixins/styleMixin'; import styleMixin from '../../../mixins/styleMixin';
import StationStatusBadge from '../../Global/StationStatusBadge.vue'; import StationStatusBadge from '../../Global/StationStatusBadge.vue';
import { ActiveScenery } from '../../../store/typings'; import { ActiveScenery } from '../../../typings/common';
import donatorMixin from '../../../mixins/donatorMixin'; import { useApiStore } from '../../../store/apiStore';
export default defineComponent({ export default defineComponent({
mixins: [styleMixin, dateMixin, routerMixin, donatorMixin], mixins: [styleMixin, dateMixin, routerMixin],
data() {
return {
apiStore: useApiStore()
};
},
props: { props: {
onlineScenery: { onlineScenery: {
type: Object as PropType<ActiveScenery>, type: Object as PropType<ActiveScenery>,
@@ -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>
&bull; <b>{{ $t('controls.title') }}:</b>
{{ $t(`controls.${station.generalInfo.controlType}`) }}
</span>
<span>
&bull; <b>{{ $t('signals.title') }}:</b>
{{ $t(`signals.${station.generalInfo.signalType}`) }}
</span>
<span v-if="station.generalInfo.lines">
&bull; <b>{{ $t('scenery.lines-title') }}:</b> {{ station.generalInfo.lines }}
</span>
<span v-if="station.generalInfo.project">
&bull; <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">
&bull; <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}`)
"
v-html="getControlTypeAbbrev(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,25 +83,16 @@
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>
<script lang="ts"> <script lang="ts">
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import stationInfoMixin from '../../../mixins/stationInfoMixin';
import styleMixin from '../../../mixins/styleMixin'; import styleMixin from '../../../mixins/styleMixin';
import Station from '../../../scripts/interfaces/Station'; import { Station } from '../../../typings/common';
export default defineComponent({ export default defineComponent({
mixins: [stationInfoMixin, styleMixin], mixins: [styleMixin],
props: { props: {
station: { station: {
type: Object as PropType<Station> type: Object as PropType<Station>
@@ -103,7 +102,8 @@ 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;
justify-content: center; justify-content: center;
@@ -111,6 +111,7 @@ export default defineComponent({
margin: 1em; margin: 1em;
} }
.icon-info { .icon-info {
display: flex; display: flex;
justify-content: center; justify-content: center;
@@ -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">
<b>{{ $t('scenery.one-way-routes') }}</b> <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>
<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,24 @@
</div> </div>
<div class="routes two-way" v-if="twoWayRoutes.length > 0"> <div class="routes two-way" v-if="twoWayRoutes.length > 0">
<b>{{ $t('scenery.two-way-routes') }}</b> <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>
<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 v-if="route.routeSpeed" class="speed">{{ route.routeSpeed }}</span>
<span :class="{ 'no-catenary': !route.isElectric, internal: route.isInternal }">{{ <span v-if="route.routeLength" class="length">
route.routeName {{ (route.routeLength / 1000).toFixed(1) + 'km' }}
}}</span>
<span v-if="route.routeSpeed" class="speed">
{{
activeShowLength.includes(route.routeName)
? route.routeLength + 'm'
: route.routeSpeed
}}
</span> </span>
<span v-if="route.isRouteSBL" class="sbl">SBL</span> <span v-if="route.isRouteSBL" class="sbl">SBL</span>
</li> </li>
@@ -52,7 +56,8 @@
<script lang="ts"> <script lang="ts">
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import Station from '../../../scripts/interfaces/Station'; import { Station } from '../../../typings/common';
import StorageManager from '../../../managers/storageManager';
export default defineComponent({ export default defineComponent({
props: { props: {
@@ -62,27 +67,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)) ?? []
);
} }
} }
}); });
@@ -92,20 +120,26 @@ export default defineComponent({
.info-routes { .info-routes {
display: flex; display: flex;
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;
@@ -121,7 +155,7 @@ ul.routes-list {
-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 +172,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;
@@ -8,7 +8,7 @@
<transition-group name="spawns-anim" tag="ul"> <transition-group name="spawns-anim" tag="ul">
<li <li
class="badge spawn badge-none" class="badge badge-none"
v-if="!onlineScenery || onlineScenery.spawns.length == 0" v-if="!onlineScenery || onlineScenery.spawns.length == 0"
key="no-spawns" key="no-spawns"
> >
@@ -16,13 +16,13 @@
</li> </li>
<li <li
class="badge spawn" class="badge spawn-badge"
v-for="(spawn, i) in sortedSpawns" v-for="(spawn, i) in sortedSpawns"
:key="spawn.spawnName + onlineScenery?.dispatcherName + i" :key="spawn.spawnName + onlineScenery?.dispatcherName + i"
:data-electrified="spawn.isElectrified" :data-electrified="spawn.isElectrified"
> >
<span class="spawn_name">{{ spawn.spawnName }}</span> <span class="name">{{ spawn.spawnName }}</span>
<span class="spawn_length">{{ spawn.spawnLength }}m</span> <span class="length">{{ spawn.spawnLength }}m</span>
</li> </li>
</transition-group> </transition-group>
</section> </section>
@@ -30,7 +30,7 @@
<script lang="ts"> <script lang="ts">
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import { ActiveScenery } from '../../../store/typings'; import { ActiveScenery } from '../../../typings/common';
export default defineComponent({ export default defineComponent({
props: { props: {
@@ -53,25 +53,10 @@ export default defineComponent({
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@import '../../../styles/variables.scss';
ul { ul {
position: relative; position: relative;
} }
.spawn {
color: white;
&_length {
background-color: #404040;
color: #cfcfcf;
}
&[data-electrified='true'] > &_name {
background-color: #007599;
}
}
.spawns-anim { .spawns-anim {
&-move, &-move,
&-enter-active, &-enter-active,
@@ -1,83 +0,0 @@
<template>
<section class="info-stats" :class="!station.onlineInfo ? 'no-stats' : ''">
<span class="likes">
<img src="/images/icon-like" alt="Likes count icon" />
<span>{{ station.onlineInfo?.dispatcherRate || '0' }}</span>
</span>
<span class="users">
<img src="/images/icon-user" alt="Users count icon" />
<span>{{ station.onlineInfo?.currentUsers || '0' }}</span>
/
<span>{{ station.onlineInfo?.maxUsers || '0' }}</span>
</span>
<span class="spawns">
<img src="/images/icon-spawn" alt="Spawns count icon" />
<span>{{ station.onlineInfo?.spawns.length || '0' }}</span>
</span>
<span class="schedules">
<img src="/images/icon-timetable" alt="Timetables count icon" />
<span>
<span style="color: #eee">{{ station.onlineInfo?.scheduledTrains?.length || '0' }}</span>
/
<span style="color: #bbb"
>{{
station.onlineInfo?.scheduledTrains?.filter((train) => train.stopInfo.confirmed)
.length || '0'
}}
</span>
</span>
</span>
</section>
</template>
<script lang="ts">
import { PropType, defineComponent } from 'vue';
import Station from '../../../scripts/interfaces/Station';
export default defineComponent({
props: {
station: {
type: Object as PropType<Station>,
required: true
}
}
});
</script>
<style lang="scss" scoped>
@import '../../../styles/variables.scss';
.info-stats {
padding: 1rem 0;
display: flex;
flex-wrap: wrap;
justify-content: center;
font-size: 1.65em;
&.no-stats {
opacity: 0.5;
}
& > span {
display: flex;
align-items: center;
margin: 0.3em;
}
.likes,
.spawns {
color: $accentCol;
}
span > img {
width: 1.2em;
margin-right: 0.5em;
}
}
</style>
@@ -13,16 +13,35 @@
</li> </li>
<li <li
v-for="train in onlineScenery?.stationTrains" v-for="{ train, status } in stationTrains"
class="badge user" class="badge user"
:class="train.stopStatus" :key="train.id"
:key="train.trainId" :data-status="status"
tabindex="0"
@click.prevent="selectModalTrain(train.trainId, $event.currentTarget)"
@keydown.enter="selectModalTrain(train.trainId, $event.currentTarget)"
> >
<span class="user_train">{{ train.trainNo }}</span> <router-link :to="train.driverRouteLocation">
<span class="user_name">{{ train.driverName }}</span> <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>
</li> </li>
</transition-group> </transition-group>
</section> </section>
@@ -30,17 +49,57 @@
<script lang="ts"> <script lang="ts">
import { PropType, defineComponent } from 'vue'; import { PropType, defineComponent } from 'vue';
import modalTrainMixin from '../../../mixins/modalTrainMixin';
import routerMixin from '../../../mixins/routerMixin'; import routerMixin from '../../../mixins/routerMixin';
import { ActiveScenery } from '../../../store/typings'; import { ActiveScenery, Station } from '../../../typings/common';
import { getTrainStopStatus } from '../utils';
import { useMainStore } from '../../../store/mainStore';
export default defineComponent({ export default defineComponent({
mixins: [routerMixin, modalTrainMixin], mixins: [routerMixin],
props: { props: {
onlineScenery: { onlineScenery: {
type: Object as PropType<ActiveScenery>, type: Object as PropType<ActiveScenery>,
required: false required: false
},
station: {
type: Object as PropType<Station>
}
},
data() {
return {
mainStore: useMainStore()
};
},
computed: {
stationTrains() {
if (!this.onlineScenery) return;
const name = this.station?.generalInfo?.checkpoints[0] ?? this.onlineScenery.name;
return this.onlineScenery.stationTrains.map((train) => {
const stop = train.timetableData?.followingStops.find(
(stop) =>
stop.stopNameRAW.toLowerCase() == name.toLowerCase() ||
this.station?.generalInfo?.checkpoints.includes(stop.stopNameRAW)
);
const sceneryName =
train.currentStationName.indexOf('.sc') != -1
? train.currentStationName.split(' ').slice(0, -1).join(' ')
: train.currentStationName;
const status = stop
? getTrainStopStatus(stop, sceneryName, this.onlineScenery!.name)
: 'no-timetable';
return {
train,
status
};
});
} }
} }
}); });
@@ -63,46 +122,45 @@ ul {
} }
.user { .user {
cursor: pointer; &[data-status='no-timetable'] .user_train {
&_train {
color: black;
background-color: $no-timetable;
transition: background-color 200ms;
-ms-transition: background-color 200ms;
-webkit-transition: background-color 200ms;
}
&.no-timetable .user_train {
background-color: $no-timetable; background-color: $no-timetable;
} }
&.departed > &_train { &[data-status='departed'] .user_train {
background-color: $departed; background-color: $departed;
} }
&.stopped > &_train { &[data-status='stopped'] .user_train {
background-color: $stopped; background-color: $stopped;
} }
&.online > &_train { &[data-status='online'] .user_train {
background-color: $online; background-color: $online;
} }
&.terminated > &_train { &[data-status='terminated'] .user_train {
background-color: $terminated; background-color: $terminated;
} }
&.disconnected > &_train { &[data-status='disconnected'] .user_train {
background-color: $disconnected; background-color: $disconnected;
} }
&.offline { &[data-status='offline'] {
background: firebrick; background: firebrick;
pointer-events: none; pointer-events: none;
} }
} }
.user_train {
color: black;
background-color: $no-timetable;
transition: background-color 200ms;
-ms-transition: background-color 200ms;
-webkit-transition: background-color 200ms;
}
.users-anim { .users-anim {
&-move, &-move,
&-enter-active, &-enter-active,
+151 -115
View File
@@ -14,13 +14,9 @@
</span> </span>
<span class="header_links" v-if="station"> <span class="header_links" v-if="station">
<!-- <a <a :href="pragotronHref" target="_blank" :title="$t('scenery.pragotron-link')">
:href="`https://pragotron-td2.web.app/board?name=${station.name}`"
target="_blank"
:title="$t('scenery.pragotron-link')"
>
<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" :title="$t('scenery.tablice-link')">
<img src="/images/icon-tablice.ico" alt="icon-tablice" /> <img src="/images/icon-tablice.ico" alt="icon-tablice" />
@@ -29,26 +25,23 @@
</h3> </h3>
<div class="timetable-checkpoints" v-if="station?.generalInfo?.checkpoints"> <div class="timetable-checkpoints" v-if="station?.generalInfo?.checkpoints">
<span v-for="(cp, i) in station.generalInfo.checkpoints" :key="i"> <template v-for="(ch, i) in station.generalInfo.checkpoints" :key="i">
{{ (i > 0 && '&bull;') || '' }} <template v-if="i > 0">&bull;</template>
<router-link
<button class="checkpoint-item"
:key="cp" :class="{ current: chosenCheckpoint === ch }"
class="checkpoint_item" :to="`/scenery?station=${station.name}&checkpoint=${ch}`"
:class="{ current: chosenCheckpoint === cp }" >{{ ch }}</router-link
@click="setCheckpoint(cp)"
> >
{{ cp }} </template>
</button>
</span>
</div> </div>
</div> </div>
<div class="timetable-list"> <div class="timetable-list">
<transition-group name="list-anim"> <transition-group name="list-anim">
<div <div
v-if="apiStore.dataStatuses.connection == 0 && sceneryTimetables.length == 0"
style="padding-bottom: 5em" style="padding-bottom: 5em"
v-if="apiStore.dataStatuses.connection == 0 && computedScheduledTrains.length == 0"
key="list-loading" key="list-loading"
> >
<Loading /> <Loading />
@@ -56,7 +49,7 @@
<span <span
class="timetable-item empty" class="timetable-item empty"
v-else-if="computedScheduledTrains.length == 0 && !onlineScenery" v-else-if="sceneryTimetables.length == 0 && !onlineScenery"
key="list-offline" key="list-offline"
> >
{{ $t('scenery.offline') }} {{ $t('scenery.offline') }}
@@ -64,68 +57,72 @@
<div <div
class="timetable-item empty" class="timetable-item empty"
v-else-if="computedScheduledTrains.length == 0" v-else-if="sceneryTimetables.length == 0"
key="list-no-timetables" key="list-no-timetables"
> >
{{ $t('scenery.no-timetables') }} {{ $t('scenery.no-timetables') }}
</div> </div>
<div <router-link
class="timetable-item" class="timetable-item"
v-else v-else
v-for="scheduledTrain in computedScheduledTrains" v-for="(row, i) in sceneryTimetables"
:key="scheduledTrain.trainId" :key="row.train.id + i"
tabindex="0" tabindex="0"
@click.prevent.stop="selectModalTrain(scheduledTrain.trainId, $event.currentTarget)" :to="row.train.driverRouteLocation"
@keydown.enter.prevent="selectModalTrain(scheduledTrain.trainId, $event.currentTarget)"
> >
<span class="timetable-general"> <span class="timetable-general">
<span class="general-info"> <span class="general-info">
<span class="info-number"> <div class="info-train">
<strong>{{ scheduledTrain.category }}</strong> <b
{{ scheduledTrain.trainNo }} data-tooltip-type="BaseTooltip"
:data-tooltip-content="getCategoryExplanation(row.train.timetableData!.category)"
class="text--primary tooltip-help"
>
{{ row.train.timetableData!.category }}
</b>
<span>&nbsp;</span>
<b>{{ row.train.trainNo }}</b>
<span>&nbsp;&bull;&nbsp;</span>
<span>{{ row.train.driverName }}</span>
<span <span
v-if="scheduledTrain.stopInfo.comments" v-if="row.checkpointStop.comments"
:title="scheduledTrain.stopInfo.comments" data-tooltip-type="BaseTooltip"
:data-tooltip-content="row.checkpointStop.comments"
> >
<img src="/images/icon-warning.svg" /> <img src="/images/icon-warning.svg" />
</span> </span>
</span>
&nbsp;|&nbsp;
<span>
{{ scheduledTrain.driverName }}
</span>
<div class="info-route">
<strong>{{ scheduledTrain.beginsAt }} - {{ scheduledTrain.terminatesAt }}</strong>
</div> </div>
<ScheduledTrainStatus :scheduledTrain="scheduledTrain" /> <div class="info-route">
<strong>{{ row.train.timetableData!.route.replace('|', ' - ') }}</strong>
</div>
<ScheduledTrainStatus :sceneryTimetableRow="row" />
</span> </span>
</span> </span>
<span class="timetable-schedule"> <span class="timetable-schedule">
<span class="schedule-arrival"> <span class="schedule-arrival">
<span class="arrival-time begins" v-if="scheduledTrain.stopInfo.beginsHere"> <span class="arrival-time begins" v-if="row.checkpointStop.beginsHere">
{{ $t('timetables.begins') }} {{ $t('timetables.begins') }}
</span> </span>
<span class="arrival-time" v-else> <span class="arrival-time" v-else>
<div v-if="scheduledTrain.stopInfo.arrivalDelay == 0"> <div v-if="row.checkpointStop.arrivalDelay == 0">
<span>{{ timestampToString(scheduledTrain.stopInfo.arrivalTimestamp) }}</span> <span>{{ timestampToString(row.checkpointStop.arrivalTimestamp) }}</span>
</div> </div>
<div v-else> <div v-else>
<div> <div>
<s style="margin-right: 0.2em" class="text--grayed">{{ <s style="margin-right: 0.2em" class="text--grayed">{{
timestampToString(scheduledTrain.stopInfo.arrivalTimestamp) timestampToString(row.checkpointStop.arrivalTimestamp)
}}</s> }}</s>
</div> </div>
<span> <span>
{{ timestampToString(scheduledTrain.stopInfo.arrivalRealTimestamp) }} {{ timestampToString(row.checkpointStop.arrivalRealTimestamp) }}
({{ scheduledTrain.stopInfo.arrivalDelay > 0 ? '+' : '' ({{ row.checkpointStop.arrivalDelay > 0 ? '+' : ''
}}{{ scheduledTrain.stopInfo.arrivalDelay }}) }}{{ row.checkpointStop.arrivalDelay }})
</span> </span>
</div> </div>
</span> </span>
@@ -133,47 +130,45 @@
<span class="schedule-stop"> <span class="schedule-stop">
<span class="stop-connection"> <span class="stop-connection">
{{ scheduledTrain.arrivingLine }} {{ row.currentElement.arrivalRouteExt }}
</span> </span>
<span class="stop-time"> <span class="stop-time">
{{ scheduledTrain.stopInfo.stopTime || '' }} {{ row.checkpointStop.stopTime || '' }}
{{ {{ row.checkpointStop.stopTime ? row.checkpointStop.stopType || 'pt' : '' }}
scheduledTrain.stopInfo.stopTime ? scheduledTrain.stopInfo.stopType || 'pt' : ''
}}
</span> </span>
<span class="stop-connection"> <span class="stop-connection">
{{ scheduledTrain.departureLine }} {{ row.currentElement.departureRouteExt }}
</span> </span>
</span> </span>
<span class="schedule-departure"> <span class="schedule-departure">
<span class="departure-time terminates" v-if="scheduledTrain.stopInfo.terminatesHere"> <span class="departure-time terminates" v-if="row.checkpointStop.terminatesHere">
{{ $t('timetables.terminates') }} {{ $t('timetables.terminates') }}
</span> </span>
<span class="departure-time" v-else> <span class="departure-time" v-else>
<div v-if="scheduledTrain.stopInfo.departureDelay == 0"> <div v-if="row.checkpointStop.departureDelay == 0">
<span>{{ timestampToString(scheduledTrain.stopInfo.departureTimestamp) }}</span> <span>{{ timestampToString(row.checkpointStop.departureTimestamp) }}</span>
</div> </div>
<div v-else> <div v-else>
<div> <div>
<s style="margin-right: 0.2em" class="text--grayed">{{ <s style="margin-right: 0.2em" class="text--grayed">{{
timestampToString(scheduledTrain.stopInfo.departureTimestamp) timestampToString(row.checkpointStop.departureTimestamp)
}}</s> }}</s>
</div> </div>
<span> <span>
{{ timestampToString(scheduledTrain.stopInfo.departureRealTimestamp) }} {{ timestampToString(row.checkpointStop.departureRealTimestamp) }}
({{ scheduledTrain.stopInfo.departureDelay > 0 ? '+' : '' ({{ row.checkpointStop.departureDelay > 0 ? '+' : ''
}}{{ scheduledTrain.stopInfo.departureDelay }}) }}{{ row.checkpointStop.departureDelay }})
</span> </span>
</div> </div>
</span> </span>
</span> </span>
</span> </span>
</div> </router-link>
</transition-group> </transition-group>
</div> </div>
</section> </section>
@@ -186,19 +181,20 @@ import { useRoute } from 'vue-router';
import Loading from '../Global/Loading.vue'; import Loading from '../Global/Loading.vue';
import dateMixin from '../../mixins/dateMixin'; import dateMixin from '../../mixins/dateMixin';
import routerMixin from '../../mixins/routerMixin'; import routerMixin from '../../mixins/routerMixin';
import Station from '../../scripts/interfaces/Station'; import trainCategoryMixin from '../../mixins/trainCategoryMixin';
import { useMainStore } from '../../store/mainStore'; import { useMainStore } from '../../store/mainStore';
import modalTrainMixin from '../../mixins/modalTrainMixin';
import ScheduledTrainStatus from './ScheduledTrainStatus.vue';
import { ActiveScenery } from '../../store/typings';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
import ScheduledTrainStatus from './ScheduledTrainStatus.vue';
import { SceneryTimetableRow } from './typings';
import { ActiveScenery, Station } from '../../typings/common';
import { getTrainStopStatus, stopStatusPriority } from './utils';
export default defineComponent({ export default defineComponent({
name: 'SceneryTimetable', name: 'SceneryTimetable',
components: { Loading, ScheduledTrainStatus }, components: { Loading, ScheduledTrainStatus },
mixins: [dateMixin, routerMixin, modalTrainMixin], mixins: [dateMixin, routerMixin, trainCategoryMixin],
props: { props: {
station: { station: {
@@ -213,12 +209,14 @@ export default defineComponent({
listOpen: false listOpen: false
}), }),
mounted() { activated() {
this.loadSelectedOption(); this.loadSelectedOption();
}, },
activated() { watch: {
this.loadSelectedOption(); currentURL() {
this.loadSelectedOption();
}
}, },
setup(props) { setup(props) {
@@ -229,9 +227,10 @@ export default defineComponent({
const mainStore = useMainStore(); const mainStore = useMainStore();
const chosenCheckpoint = ref( const chosenCheckpoint = ref(
props.station?.generalInfo?.checkpoints?.length == 0 props.station?.generalInfo?.checkpoints[0] ??
? '' props.station?.name ??
: props.station?.generalInfo?.checkpoints[0] ?? null route.query['station']?.toString() ??
''
); );
return { return {
@@ -250,27 +249,54 @@ export default defineComponent({
return url; return url;
}, },
computedScheduledTrains() { pragotronHref() {
if (!this.station) return []; let url = `https://pragotron-td2.web.app/board?name=${this.station!.name}&region=${this.mainStore.region.id}`;
if (this.chosenCheckpoint) url += `&checkpoint=${this.chosenCheckpoint}`;
return ( return url;
this.onlineScenery?.scheduledTrains },
?.filter(
(train) =>
train.checkpointName.toLocaleLowerCase() ==
(this.chosenCheckpoint || this.station!.name).toLocaleLowerCase() &&
train.region == this.mainStore.region.id
)
.sort((a, b) => {
if (a.stopStatusID > b.stopStatusID) return 1;
if (a.stopStatusID < b.stopStatusID) return -1;
if (a.stopInfo.arrivalTimestamp > b.stopInfo.arrivalTimestamp) return 1; sceneryTimetables(): SceneryTimetableRow[] {
if (a.stopInfo.arrivalTimestamp < b.stopInfo.arrivalTimestamp) return -1; if (!this.onlineScenery) return [];
return a.stopInfo.departureTimestamp > b.stopInfo.departureTimestamp ? 1 : -1; const sceneryName = this.$route.query['station']?.toString().replace(/_/g, ' ') ?? '';
}) || []
); return this.onlineScenery.scheduledTrains
.filter(
(ct) =>
// ct.timetablePathElement.stationName == sceneryName &&
ct.train.region == this.mainStore.region.id &&
this.chosenCheckpoint &&
ct.checkpointStop.stopNameRAW.toLowerCase() == this.chosenCheckpoint.toLowerCase()
)
.map((ct) => {
const trainStopStatus = getTrainStopStatus(
ct.checkpointStop,
ct.train.currentStationName,
sceneryName
);
return {
checkpointStop: ct.checkpointStop,
train: ct.train,
prevElement: ct.previousSceneryElement,
nextElement: ct.nextSceneryElement,
currentElement: ct.timetablePathElement,
status: trainStopStatus
};
})
.sort((a, b) => {
if (stopStatusPriority.indexOf(a.status) - stopStatusPriority.indexOf(b.status) < 0)
return -1;
if (stopStatusPriority.indexOf(a.status) - stopStatusPriority.indexOf(b.status) > 0)
return 1;
if (a.checkpointStop.arrivalTimestamp > b.checkpointStop.arrivalTimestamp) return 1;
if (a.checkpointStop.arrivalTimestamp < b.checkpointStop.arrivalTimestamp) return -1;
return a.checkpointStop.departureTimestamp > b.checkpointStop.departureTimestamp ? 1 : -1;
});
} }
}, },
@@ -278,7 +304,19 @@ export default defineComponent({
loadSelectedOption() { loadSelectedOption() {
if (!this.station) return; if (!this.station) return;
this.chosenCheckpoint = this.station.generalInfo?.checkpoints[0] ?? this.station.name; if (!this.station.generalInfo) {
this.chosenCheckpoint = this.station.name;
return;
}
const queryCheckpoint = this.$route.query['checkpoint']?.toString();
this.chosenCheckpoint =
this.station.generalInfo.checkpoints.find(
(ch) => ch.toLocaleLowerCase() === queryCheckpoint?.toLocaleLowerCase()
) ??
this.station.generalInfo.checkpoints[0] ??
this.station.name;
}, },
setCheckpoint(cp: string) { setCheckpoint(cp: string) {
@@ -289,9 +327,8 @@ 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';
.scenery-timetable { .scenery-timetable {
height: 100%; height: 100%;
@@ -348,7 +385,6 @@ export default defineComponent({
background: #353535; background: #353535;
cursor: pointer;
z-index: 10; z-index: 10;
&.empty { &.empty {
@@ -378,51 +414,51 @@ export default defineComponent({
} }
} }
.timetable-list {
position: relative;
}
.timetable-checkpoints { .timetable-checkpoints {
display: flex; display: flex;
justify-content: center; justify-content: center;
gap: 0.5em;
flex-wrap: wrap; flex-wrap: wrap;
font-size: 1.1em; font-size: 1.1em;
margin-top: 0.5em; margin-top: 0.5em;
}
button.checkpoint_item { .checkpoint-item {
color: #aaa; color: #aaa;
display: inline; display: inline;
&:hover {
color: white;
} }
.checkpoint_item.current { &.current {
font-weight: bold; font-weight: bold;
color: $accentCol; color: var(--clr-primary);
} }
} }
.timetable-list {
position: relative;
}
.general-info { .general-info {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
.info-number { .info-number {
color: $accentCol; color: var(--clr-primary);
} }
.info-route { .info-route {
width: 100%; width: 100%;
} }
.g-tooltip > .content {
z-index: 100;
color: white;
left: 110%;
}
img { img {
width: 1.1em; height: 0.9em;
vertical-align: middle;
margin: 0 0.25em;
} }
} }
@@ -448,7 +484,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';
@@ -465,7 +501,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;
} }
@@ -1,69 +1,97 @@
<template> <template>
<!-- WIP --> <div class="scenery-timetables-history">
<!-- <div class="top-filters"> <div class="history-modes">
<button class="btn btn--option">ROZPOCZYNA BIEG</button> <button
class="btn btn--option"
<button class="btn btn--option">PRZEZ</button> v-for="mode in historyModeList"
:key="mode"
<button class="btn btn--option">KOŃCZY BIEG</button> :class="{ checked: checkedHistoryMode == mode }"
</div> --> @click="checkHistoryMode(mode)"
>
<section class="scenery-table-section"> {{ $t(`scenery.timetable-${mode}`) }}
<Loading v-if="dataStatus != DataStatus.Loaded" /> </button>
<div class="no-history" v-else-if="historyList.length == 0">
{{ $t('scenery.history-list-empty') }}
</div> </div>
<table class="scenery-history-table" v-else> <div class="history-wrapper">
<thead> <Loading v-if="dataStatus != DataStatus.Loaded" />
<th>{{ $t('scenery.timetables-history-id') }}</th>
<th>{{ $t('scenery.timetables-history-number') }}</th>
<th>{{ $t('scenery.timetables-history-route') }}</th>
<th>{{ $t('scenery.timetables-history-driver') }}</th>
<th>{{ $t('scenery.timetables-history-author') }}</th>
<th>{{ $t('scenery.timetables-history-date') }}</th>
</thead>
<tbody> <div v-else-if="historyList.length == 0" class="no-history">
<tr v-for="historyItem in historyList" :key="historyItem.id"> {{ $t('scenery.history-list-empty') }}
<td> </div>
<router-link :to="`/journal/timetables?search-train=%23${historyItem.id}`">
#{{ historyItem.id }}
</router-link>
</td>
<td>
<b class="text--primary">{{ historyItem.trainCategoryCode }}</b> <br />
{{ historyItem.trainNo }}
</td>
<td>{{ historyItem.route.replace('|', ' -> ') }}</td>
<td>
<router-link :to="`/journal/timetables?search-driver=${historyItem.driverName}`">
{{ historyItem.driverName }}
</router-link>
</td>
<td> <div v-else class="journal-list">
<router-link <div v-for="timetableHistory in historyList" :key="timetableHistory.id">
v-if="historyItem.authorName" <span>
:to="`/journal/timetables?search-dispatcher=${historyItem.authorName}`" <div>
>{{ historyItem.authorName }} <span
</router-link> class="timetable-status-indicator"
<i v-else>{{ $t('scenery.timetable-author-unknown') }}</i> :data-terminated="timetableHistory.terminated"
</td> :data-fulfilled="timetableHistory.fulfilled"
<td> >
<b>{{ localeDay(historyItem.beginDate, $i18n.locale) }}</b> &ofcir;
{{ localeTime(historyItem.beginDate, $i18n.locale) }} </span>
</td> #{{ timetableHistory.id }} |
</tr> <b class="text--primary">{{ timetableHistory.trainCategoryCode }}</b>
</tbody> {{ timetableHistory.trainNo }}
</table> {{ timetableHistory.route.replace('|', ' &Rightarrow; ') }}
</section> </div>
<div class="bottom-info"> <div class="text--grayed">
<button class="btn btn--option" v-if="historyList.length > 0" @click="navigateToHistory()"> <span>
{{ $t('scenery.bottom-info') }} {{ $t('scenery.timetable-issued-date') }}
</button> <b>
{{
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>
</span>
<span>
{{ $t('scenery.timetable-issued-for') }}
<b>
<router-link
:to="`/journal/timetables?search-driver=${timetableHistory.driverName}`"
>
{{ timetableHistory.driverName }}
</router-link>
</b>
</span>
</div>
</span>
<button
@click="
navigateTo(`/journal/timetables`, {
'search-train': `#${timetableHistory.id}`
})
"
>
<img src="/images/icon-back.svg" alt="icon navigate to timetable" />
</button>
</div>
</div>
</div>
<div class="bottom-info">
<button class="btn btn--option" v-if="historyList.length > 0" @click="navigateToHistory()">
{{ $t('scenery.bottom-info') }}
</button>
</div>
</div> </div>
</template> </template>
@@ -71,17 +99,19 @@
import { defineComponent, PropType } from 'vue'; import { defineComponent, PropType } from 'vue';
import dateMixin from '../../mixins/dateMixin'; import dateMixin from '../../mixins/dateMixin';
import Station from '../../scripts/interfaces/Station';
import Loading from '../Global/Loading.vue'; import Loading from '../Global/Loading.vue';
import listObserverMixin from '../../mixins/listObserverMixin';
import { ActiveScenery } from '../../store/typings';
import { API } from '../../typings/api'; import { API } from '../../typings/api';
import { Status } from '../../typings/common'; import { ActiveScenery, Station, Status } from '../../typings/common';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
import routerMixin from '../../mixins/routerMixin';
import { useMainStore } from '../../store/mainStore';
const historyModeList = ['via', 'issuedFrom', 'terminatingAt'] as const;
type HistoryMode = (typeof historyModeList)[number];
export default defineComponent({ export default defineComponent({
name: 'SceneryTimetablesHistory', name: 'SceneryTimetablesHistory',
mixins: [dateMixin, listObserverMixin], mixins: [dateMixin, routerMixin],
props: { props: {
station: { station: {
type: Object as PropType<Station> type: Object as PropType<Station>
@@ -94,9 +124,14 @@ export default defineComponent({
data() { data() {
return { return {
historyList: [] as API.TimetableHistory.Response, historyList: [] as API.TimetableHistory.Response,
historyModeList,
apiStore: useApiStore(), apiStore: useApiStore(),
mainStore: useMainStore(),
dataStatus: Status.Data.Loading, dataStatus: Status.Data.Loading,
DataStatus: Status.Data DataStatus: Status.Data,
checkedHistoryMode: 'via' as HistoryMode
}; };
}, },
@@ -106,17 +141,22 @@ export default defineComponent({
methods: { methods: {
async fetchAPIData() { async fetchAPIData() {
if (!this.station && !this.onlineScenery) { const stationName = this.$route.query['station'];
if (!stationName) {
this.historyList = [];
this.dataStatus = Status.Data.Loaded; this.dataStatus = Status.Data.Loaded;
return; return;
} }
const requestFilters: Record<string, any> = {};
requestFilters[this.checkedHistoryMode] = stationName.toString();
requestFilters.countLimit = 30;
try { try {
const response: API.TimetableHistory.Response = await ( const response: API.TimetableHistory.Response = await (
await this.apiStore.client!.get('api/getTimetables', { await this.apiStore.client!.get('api/getTimetables', {
params: { params: requestFilters
issuedFrom: this.station?.name || this.onlineScenery?.name
}
}) })
).data; ).data;
@@ -128,11 +168,17 @@ export default defineComponent({
} }
}, },
checkHistoryMode(mode: HistoryMode) {
this.checkedHistoryMode = mode;
this.dataStatus = Status.Data.Loading;
this.fetchAPIData();
},
navigateToHistory() { navigateToHistory() {
this.$router.push({ this.$router.push({
path: '/journal/timetables', path: '/journal/timetables',
query: { query: {
'search-issuedFrom': this.station?.name || this.onlineScenery?.name [`search-${this.checkedHistoryMode}`]: this.station?.name || this.onlineScenery?.name
} }
}); });
} }
@@ -142,16 +188,69 @@ 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';
.top-filters { .scenery-timetables-history {
height: 100%;
overflow: auto;
display: grid;
gap: 1em;
grid-template-rows: auto 1fr 40px;
}
.history-wrapper {
position: relative;
overflow: auto;
}
.history-modes {
display: flex; display: flex;
justify-content: center; justify-content: center;
flex-wrap: wrap;
gap: 0.5em; gap: 0.5em;
padding: 0.25em;
button { button {
padding: 0.5em; padding: 0.35em;
min-width: 120px;
}
}
.journal-list {
display: flex;
flex-direction: column;
gap: 0.5em;
text-align: left;
}
.journal-list > div {
display: flex;
justify-content: space-between;
align-items: center;
padding: 0.5em;
background-color: #2b2b2b;
line-height: 1.5em;
}
.journal-list > div > button > img {
width: 2em;
transform: rotate(180deg);
}
.timetable-status-indicator {
&[data-fulfilled='true'] {
color: lightgreen;
}
&[data-terminated='false'] {
color: lightblue;
}
&[data-terminated='true'][data-fulfilled='false'] {
color: lightcoral;
} }
} }
</style> </style>
@@ -1,8 +1,10 @@
<template> <template>
<div class="general-status"> <div class="general-status">
<span <span
:class="computedScheduledTrain.stopStatus" :class="computedScheduledTrain.status"
:title="computedScheduledTrain.stopStatusDescription" data-tooltip-type="HtmlTooltip"
:data-tooltip-content="computedScheduledTrain.stopStatusDescription"
@click.prevent="() => {}"
> >
{{ computedScheduledTrain.stopStatusIndicator }} {{ computedScheduledTrain.stopStatusIndicator }}
</span> </span>
@@ -11,68 +13,75 @@
<script lang="ts"> <script lang="ts">
import { defineComponent, PropType } from 'vue'; import { defineComponent, PropType } from 'vue';
import { ScheduledTrain, StopStatus } from '../../store/typings'; import { StopStatus } from '../../typings/common';
import { SceneryTimetableRow } from './typings';
interface ScheduledTrainComp extends ScheduledTrain {
stopStatusIndicator: string;
stopStatusDescription: string;
}
export default defineComponent({ export default defineComponent({
props: { props: {
scheduledTrain: { sceneryTimetableRow: {
type: Object as PropType<ScheduledTrain>, type: Object as PropType<SceneryTimetableRow>,
required: true required: true
} }
}, },
computed: { computed: {
computedScheduledTrain(): ScheduledTrainComp { computedScheduledTrain() {
const { prevDepartureLine, prevStationName, stopStatus, nextArrivalLine, nextStationName } = const { status, prevElement, currentElement, nextElement } = this.sceneryTimetableRow;
this.scheduledTrain;
const prevDepartureIndicator = prevDepartureLine const prevDepartureIndicator = prevElement?.departureRouteExt
? `(${prevDepartureLine}) ${prevStationName}` ? `(${prevElement.departureRouteExt}) ${prevElement.stationName}`
: '---';
const nextArrivalIndicator = nextArrivalLine
? `(${nextArrivalLine}) ${nextStationName}`
: '---'; : '---';
const nextArrivalIndicator = nextElement?.arrivalRouteExt
? `(${nextElement.arrivalRouteExt}) ${nextElement.stationName}`
: `${currentElement.stationName}`;
let stopStatusDescription = '', let stopStatusDescription = '',
stopStatusIndicator = ''; stopStatusIndicator = '';
switch (stopStatus) { switch (status) {
case StopStatus.ARRIVING: case StopStatus.ARRIVING:
stopStatusIndicator = `${this.$t('timetables.from')}: ${prevDepartureIndicator}`; stopStatusIndicator = `${this.$t('timetables.from')}: ${prevDepartureIndicator}`;
stopStatusDescription = this.$t('timetables.desc-arriving', { stopStatusDescription = this.$t('timetables.desc-arriving', {
prevStationName, prevStationName: prevElement?.stationName ?? '',
prevDepartureLine prevDepartureLine: prevElement?.departureRouteExt ?? ''
}); });
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.to')}: ${nextArrivalIndicator}`
: `${this.$t('timetables.desc-end')}`; : `${this.$t('timetables.desc-end')}`;
stopStatusDescription = nextArrivalLine stopStatusDescription = nextElement?.arrivalRouteExt
? this.$t(`timetables.desc-${stopStatus}`, { nextStationName, nextArrivalLine }) ? this.$t(`timetables.desc-${status}`, {
nextStationName: nextElement?.stationName,
nextArrivalLine: nextElement?.arrivalRouteExt
})
: ''; : '';
break; break;
case StopStatus.DEPARTED: case StopStatus.DEPARTED:
stopStatusIndicator = `${this.$t('timetables.to')}: ${nextArrivalIndicator}`; stopStatusIndicator = `${this.$t('timetables.to')}: ${nextArrivalIndicator}`;
stopStatusDescription = this.$t('timetables.desc-departed', {
nextStationName, if (!nextElement?.stationName) {
nextArrivalLine stopStatusDescription = this.$t('timetables.desc-departed-ends', {
}); nextStationName: currentElement.stationName
});
} else {
stopStatusDescription = this.$t('timetables.desc-departed', {
nextStationName: nextElement?.stationName ?? currentElement.stationName,
nextArrivalLine: nextElement?.arrivalRouteExt
});
}
break; break;
case StopStatus.DEPARTED_AWAY: case StopStatus.DEPARTED_AWAY:
stopStatusIndicator = `${this.$t('timetables.to')}: ${nextArrivalIndicator}`; stopStatusIndicator = `${this.$t('timetables.to')}: ${nextArrivalIndicator}`;
stopStatusDescription = this.$t('timetables.desc-departed-away', { stopStatusDescription = this.$t('timetables.desc-departed-away', {
nextStationName, nextStationName: nextElement?.stationName,
nextArrivalLine nextArrivalLine: nextElement?.arrivalRouteExt
}); });
break; break;
@@ -85,7 +94,7 @@ export default defineComponent({
break; break;
} }
return { return {
...this.scheduledTrain, ...this.sceneryTimetableRow,
stopStatusDescription, stopStatusDescription,
stopStatusIndicator stopStatusIndicator
}; };
@@ -97,6 +106,7 @@ export default defineComponent({
<style lang="scss" scoped> <style lang="scss" scoped>
.general-status { .general-status {
margin-top: 0.5em; margin-top: 0.5em;
cursor: help;
span.arriving { span.arriving {
color: #ccc; color: #ccc;
+10
View File
@@ -0,0 +1,10 @@
import { StopStatus, TimetablePathElement, Train, TrainStop } from '../../typings/common';
export interface SceneryTimetableRow {
checkpointStop: TrainStop;
train: Train;
prevElement: TimetablePathElement | null;
nextElement: TimetablePathElement | null;
currentElement: TimetablePathElement;
status: StopStatus;
}
+42
View File
@@ -0,0 +1,42 @@
import { StopStatus, TrainStop } from '../../typings/common';
export const stopStatusPriority = [
StopStatus.ONLINE,
StopStatus.STOPPED,
StopStatus.DEPARTED,
StopStatus.ARRIVING,
StopStatus.DEPARTED_AWAY,
StopStatus.TERMINATED
];
export function getTrainStopStatus(
stopInfo: TrainStop,
currentStationName: string,
sceneryName: string
) {
if (stopInfo.terminatesHere && stopInfo.confirmed) {
return StopStatus.TERMINATED;
}
if (!stopInfo.terminatesHere && stopInfo.confirmed && currentStationName == sceneryName) {
return StopStatus.DEPARTED;
}
if (!stopInfo.terminatesHere && stopInfo.confirmed && currentStationName != sceneryName) {
return StopStatus.DEPARTED_AWAY;
}
if (currentStationName == sceneryName && !stopInfo.stopped) {
return StopStatus.ONLINE;
}
if (currentStationName == sceneryName && stopInfo.stopped) {
return StopStatus.STOPPED;
}
if (currentStationName != sceneryName) {
return StopStatus.ARRIVING;
}
return StopStatus.ONLINE;
}
+10 -19
View File
@@ -15,7 +15,6 @@
<script lang="ts"> <script lang="ts">
import { defineComponent } from 'vue'; import { defineComponent } from 'vue';
import { useStationFiltersStore } from '../../store/stationFiltersStore';
interface FilterOption { interface FilterOption {
id: string; id: string;
@@ -40,15 +39,9 @@ export default defineComponent({
emits: ['update:optionValue'], emits: ['update:optionValue'],
setup() {
return {
filterStore: useStationFiltersStore()
};
},
watch: { watch: {
'option.value'() { 'option.value'() {
this.filterStore.changeFilterValue(this.option.name, !this.option.value); // this.filterStore.changeFilterValue(this.option.name, !this.option.value);
} }
}, },
@@ -56,25 +49,23 @@ export default defineComponent({
handleDbClick(e: Event) { handleDbClick(e: Event) {
e.preventDefault(); e.preventDefault();
this.filterStore.lastClickedFilterId = this.option.id; // this.filterStore.lastClickedFilterId = this.option.id;
// this.option.value = true; // this.option.value = true;
this.$emit('update:optionValue', true); this.$emit('update:optionValue', true);
this.filterStore.inputs.options // this.filterStore.inputs.options
.filter((option) => { // .filter((option) => {
return option.section == this.option.section && option.id != this.option.id; // return option.section == this.option.section && option.id != this.option.id;
}) // })
.forEach((option) => { // .forEach((option) => {
option.value = !this.option.value; // option.value = !this.option.value;
}); // });
} }
} }
}); });
</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;
@@ -105,7 +96,7 @@ label {
} }
&:focus-visible + span { &:focus-visible + span {
outline: 1px solid $accentCol; outline: 1px solid var(--clr-primary);
} }
} }
} }
+262 -167
View File
@@ -1,61 +1,82 @@
<template> <template>
<section class="filter-card" v-click-outside="closeCard" @keydown.esc="closeCard"> <section class="filter-card" v-click-outside="closeCard" @keydown.esc="closeCard">
<div class="card_controls"> <div class="card_controls">
<button class="btn--filled btn--image" @click="toggleCard"> <button class="card-button btn--filled btn--image" @click="toggleCard">
<img class="button_icon" src="/images/icon-filter2.svg" alt="filter icon" /> <img class="button_icon" src="/images/icon-filter2.svg" alt="filter icon" />
{{ $t('options.filters') }} [F] <p>[F] {{ $t('options.filters') }}</p>
<span class="active-indicator" v-if="!filterStore.areFiltersAtDefault"></span> <span class="active-indicator" v-if="changedFilters.length != 0"></span>
</button> </button>
<label for="scenery-search">
<input
id="scenery-search"
list="sceneries"
:placeholder="$t('sceneries.scenery-search')"
@focus="preventKeyDown = true"
@blur="preventKeyDown = false"
v-model="chosenSearchScenery"
/>
<datalist id="sceneries">
<option
v-for="scenery in sortedStationList"
:key="scenery.name"
:value="scenery.name"
></option>
</datalist>
</label>
</div> </div>
<transition name="card-anim"> <transition name="card-anim">
<div class="card" v-if="isVisible" tabindex="0" ref="cardEl"> <div class="card" v-if="isVisible" ref="cardRef" @keydown.r="resetFilters">
<div class="card_content"> <div class="card_content" tabindex="0" @scroll="onScroll" ref="cardContentRef">
<div class="card_title flex">{{ $t('filters.title') }}</div> <div class="card_title flex">{{ $t('filters.title') }}</div>
<p class="card_info" v-html="$t('filters.desc')"></p> <p class="card_info" v-html="$t('filters.desc')"></p>
<div class="changed-filters" :data-active="changedFilters.length > 0">
<template v-if="changedFilters.length > 0">
{{ $t('filters.changed-filters-count') }} <b>{{ changedFilters.length }}</b>
</template>
<template v-else>{{ $t('filters.no-changed-filters') }}</template>
</div>
<section class="card_sceneries-search">
<h3 class="section-header">{{ $t('filters.sceneries-search') }}</h3>
<datalist id="sceneries">
<option
v-for="scenery in sortedStationList"
:key="scenery.name"
:value="scenery.name"
></option>
</datalist>
<form action="javascript:void(0);" @submit="handleSceneriesInput">
<input
v-model="chosenSearchScenery"
id="scenery-search"
list="sceneries"
:placeholder="$t('filters.sceneries-placeholder')"
@focus="preventKeyDown = true"
@blur="preventKeyDown = false"
/>
<button class="btn--action">{{ $t('filters.search-button-title') }}</button>
</form>
</section>
<section class="card_options"> <section class="card_options">
<div <div
class="option-section" class="option-section"
v-for="section in filterStore.inputs.optionSections" v-for="(sectionFilters, sectionKey) in filtersSections"
:key="section" :key="sectionKey"
> >
<h3 class="text--primary"> <h3 class="text--primary">
{{ $t(`filters.sections.${section}`) }} <span class="active-indicator" v-if="!areSectionFiltersDefault(sectionKey)"></span>
{{ $t(`filters.sections.${sectionKey}`) }}
<button @click="filterStore.resetSectionOptions(section)">RESET</button> <button @click="resetSectionFilters(sectionKey)">RESET</button>
</h3> </h3>
<hr /> <hr />
<div class="section-inputs"> <div class="section-filters">
<FilterOption <label
v-for="(option, i) in filterStore.inputs.options.filter( v-for="filterKey in sectionFilters"
(o) => o.section == section @dblclick="setSingleSectionFilter(sectionKey, filterKey)"
)" >
v-model:optionValue="option.value" <input
:option="option" type="checkbox"
:key="i" :checked="filters[filterKey]"
/> v-model="filters[filterKey]"
:class="sectionKey"
:name="filterKey"
:id="filterKey"
/>
<span>
{{ $t(`filters.${filterKey}`) }}
</span>
</label>
</div> </div>
</div> </div>
</section> </section>
@@ -68,50 +89,50 @@
<span>{{ <span>{{
minimumHours == 0 minimumHours == 0
? $t('filters.now') ? $t('filters.now')
: minimumHours < 8 : minimumHours < 7
? minimumHours + $t('filters.hour') ? minimumHours + $t('filters.hour')
: $t('filters.no-limit') : $t('filters.no-limit')
}}</span> }}</span>
<button class="btn--action" @click="addHour">+</button> <button class="btn--action" @click="addHour">+</button>
</span> </span>
</section> </section>
<datalist id="authors">
<option v-for="(author, i) in authors" :key="i" :value="author"></option>
</datalist>
<section class="card_authors-search"> <section class="card_authors-search">
<h3 class="section-header">{{ $t('filters.authors-search') }}</h3> <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"> <form action="javascript:void(0);" @submit="handleAuthorsInput">
<input <input
type="text" type="text"
id="author" id="author"
list="authors" list="authors"
name="authors" name="authors"
v-model="authors"
:placeholder="$t('filters.authors-placeholder')" :placeholder="$t('filters.authors-placeholder')"
v-model="authorsInputValue"
@focus="preventKeyDown = true" @focus="preventKeyDown = true"
@blur="preventKeyDown = false" @blur="preventKeyDown = false"
/> />
<button class="btn--action">{{ $t('filters.authors-button-title') }}</button> <button class="btn--action">{{ $t('filters.search-button-title') }}</button>
</form> </form>
</section> </section>
<section class="card_sliders"> <section class="card_sliders">
<div class="slider" v-for="(slider, i) in filterStore.inputs.sliders" :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"
:name="slider.name" :name="slider.id"
:id="slider.id" :id="slider.id"
:min="slider.minRange" :min="slider.minRange"
:max="slider.maxRange" :max="slider.maxRange"
v-model="slider.value" :step="slider.step"
@change="handleInput" v-model.number="filters[slider.id]"
/> />
<span class="slider-value">{{ slider.value }}</span> <span class="slider-value">{{ filters[slider.id] }}</span>
<div class="slider-content"> <div class="slider-content">
{{ $t(`filters.sliders.${slider.id}`) }} {{ $t(`filters.sliders.${slider.id}`) }}
</div> </div>
@@ -132,11 +153,11 @@
<button <button
class="btn--action" class="btn--action"
:disabled="changedFilters.length == 0"
:data-disabled="changedFilters.length == 0"
@click="resetFilters" @click="resetFilters"
:disabled="filterStore.areFiltersAtDefault"
:data-disabled="filterStore.areFiltersAtDefault"
> >
{{ $t('filters.reset') }} [R] {{ $t('filters.reset') }}
</button> </button>
<button class="btn--action" @click="closeCard">{{ $t('filters.close') }}</button> <button class="btn--action" @click="closeCard">{{ $t('filters.close') }}</button>
</div> </div>
@@ -150,48 +171,76 @@
import { defineComponent, inject } from 'vue'; import { defineComponent, inject } from 'vue';
import keyMixin from '../../mixins/keyMixin'; import keyMixin from '../../mixins/keyMixin';
import routerMixin from '../../mixins/routerMixin'; import routerMixin from '../../mixins/routerMixin';
import { useStationFiltersStore } from '../../store/stationFiltersStore';
import { useMainStore } from '../../store/mainStore'; import { useMainStore } from '../../store/mainStore';
import FilterOption from './FilterOption.vue'; import FilterOption from './FilterOption.vue';
import StorageManager from '../../managers/storageManager'; import StorageManager from '../../managers/storageManager';
import {
filtersSections,
sliderStates,
initFilters,
getChangedFilters
} from '../../managers/stationFilterManager';
import { StationFilterSection } from '../../managers/stationFilterManager';
import { computed } from 'vue';
import { watch } from 'vue';
const STORAGE_KEY = 'options_saved';
export default defineComponent({ export default defineComponent({
components: { FilterOption }, components: { FilterOption },
mixins: [keyMixin, routerMixin], mixins: [keyMixin, routerMixin],
data: () => ({ data: () => ({
saveOptions: false, saveOptions: false,
STORAGE_KEY: 'options_saved',
authorsInputValue: '', filtersSections,
sliderStates,
minimumHours: 0, minimumHours: 0,
authors: '',
currentRegion: { id: '', value: '' }, currentRegion: { id: '', value: '' },
delayInputTimer: -1, delayInputTimer: -1,
chosenSearchScenery: '' chosenSearchScenery: '',
scrollTop: 0,
lastFocusedEl: null as HTMLElement | null
}), }),
setup() { setup() {
const isVisible = inject('isFilterCardVisible'); const isVisible = inject('isFilterCardVisible');
const store = useMainStore(); const store = useMainStore();
const filterStore = useStationFiltersStore();
const filters = inject('StationsView_filters') as Record<string, any>;
const changedFilters = computed(() => getChangedFilters(filters));
// Save filters to persistent storage
watch(filters, (value) => {
if (!StorageManager.isRegistered(STORAGE_KEY)) return;
Object.keys(value).forEach((filterKey) => {
StorageManager.setValue(filterKey, filters[filterKey]);
});
});
return { return {
isVisible, isVisible,
store, store,
filterStore filters,
changedFilters
}; };
}, },
mounted() { mounted() {
this.saveOptions = StorageManager.isRegistered(this.STORAGE_KEY); this.saveOptions = StorageManager.isRegistered(STORAGE_KEY);
if (StorageManager.isRegistered('onlineFromHours') && this.saveOptions) { if (StorageManager.isRegistered('onlineFromHours') && this.saveOptions) {
this.minimumHours = StorageManager.getNumericValue('onlineFromHours'); this.minimumHours = StorageManager.getNumericValue('onlineFromHours');
this.changeNumericFilterValue('onlineFromHours', this.minimumHours);
} }
this.currentRegion = this.store.region; this.currentRegion = this.store.region;
@@ -210,7 +259,7 @@ export default defineComponent({
return true; return true;
}, },
authors() { 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) => {
@@ -225,18 +274,12 @@ export default defineComponent({
}, },
watch: { watch: {
chosenSearchScenery(value: string) {
const chosenStation = this.store.stationList.find(({ name }) => name == value);
if (chosenStation) {
this.$router.push(`/scenery?station=${chosenStation.name.replace(/ /g, '_')}`);
this.chosenSearchScenery = '';
}
},
isVisible(value: boolean) { isVisible(value: boolean) {
this.$nextTick(() => { this.$nextTick(() => {
if (value) (this.$refs['cardEl'] as HTMLDivElement).focus(); if (value) {
(this.$refs['cardContentRef'] as HTMLDivElement).scrollTop = this.scrollTop;
(this.$refs['cardContentRef'] as HTMLDivElement).focus();
}
}); });
} }
}, },
@@ -247,61 +290,81 @@ export default defineComponent({
this.isVisible = !this.isVisible; this.isVisible = !this.isVisible;
}, },
handleInput(e: Event) { onScroll(e: Event) {
const target = e.target as HTMLInputElement; this.scrollTop = (e.target as HTMLElement).scrollTop;
this.filterStore.changeFilterValue(target.name, target.value);
if (this.saveOptions) StorageManager.setStringValue(target.name, target.value);
}, },
handleAuthorsInput() { handleAuthorsInput() {
this.filterStore.changeFilterValue('authors', this.authorsInputValue); this.filters['authors'] = this.authors;
if (this.saveOptions) StorageManager.setStringValue('authors', this.authorsInputValue);
}, },
changeNumericFilterValue(name: string, value: number, saveToStorage = false) { handleSceneriesInput() {
this.filterStore.changeFilterValue(name, value); const chosenStation = this.store.stationList.find(
if (this.saveOptions && saveToStorage) StorageManager.setNumericValue(name, value); ({ name }) => name == this.chosenSearchScenery
);
if (chosenStation) {
this.$router.push(`/scenery?station=${chosenStation.name.replace(/ /g, '_')}`);
this.chosenSearchScenery = '';
this.isVisible = false;
}
}, },
subHour() { subHour() {
this.minimumHours = this.minimumHours < 1 ? 8 : this.minimumHours - 1; this.minimumHours = this.minimumHours < 1 ? 7 : this.minimumHours - 1;
this.filters['onlineFromHours'] = this.minimumHours;
this.changeNumericFilterValue('onlineFromHours', this.minimumHours, true);
}, },
addHour() { addHour() {
this.minimumHours = this.minimumHours > 7 ? 0 : this.minimumHours + 1; this.minimumHours = this.minimumHours > 6 ? 0 : this.minimumHours + 1;
this.filters['onlineFromHours'] = this.minimumHours;
this.changeNumericFilterValue('onlineFromHours', this.minimumHours, true);
}, },
saveFilters() { saveFilters() {
this.saveOptions = !this.saveOptions; this.saveOptions = !this.saveOptions;
if (!this.saveOptions) { if (!this.saveOptions) {
StorageManager.unregisterStorage(this.STORAGE_KEY); StorageManager.unregisterStorage(STORAGE_KEY);
return; return;
} }
StorageManager.registerStorage(this.STORAGE_KEY); StorageManager.registerStorage(STORAGE_KEY);
this.filterStore.inputs.options.forEach((option) => Object.keys(this.filters).forEach((filterKey) => {
StorageManager.setBooleanValue(option.name, !option.value) StorageManager.setValue(filterKey, this.filters[filterKey]);
); });
this.filterStore.inputs.sliders.forEach((slider) =>
StorageManager.setNumericValue(slider.name, slider.value)
);
}, },
resetFilters() { resetFilters() {
this.authorsInputValue = ''; if (this.preventKeyDown) return;
// Reset local model values
this.minimumHours = 0; this.minimumHours = 0;
this.changeNumericFilterValue('onlineFromHours', this.minimumHours, true); this.authors = '';
this.filterStore.resetFilters();
// Reset global filters
Object.keys(this.filters).forEach((filterKey) => {
this.filters[filterKey] = (initFilters as any)[filterKey];
});
},
areSectionFiltersDefault(sectionKey: StationFilterSection) {
return filtersSections[sectionKey].every((filterKey) => {
return this.filters[filterKey] == initFilters[filterKey];
});
},
resetSectionFilters(sectionKey: StationFilterSection) {
filtersSections[sectionKey].forEach((filterKey) => {
this.filters[filterKey] = initFilters[filterKey];
});
},
setSingleSectionFilter(sectionKey: StationFilterSection, chosenKey: string) {
filtersSections[sectionKey].forEach((filterKey) => {
if (typeof this.filters[filterKey] === 'boolean')
this.filters[filterKey] = filterKey != chosenKey;
});
}, },
closeCard() { closeCard() {
@@ -316,9 +379,9 @@ export default defineComponent({
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@import '../../styles/responsive.scss'; @use '../../styles/responsive';
@import '../../styles/card.scss'; @use '../../styles/card';
@import '../../styles/animations.scss'; @use '../../styles/animations';
h3.section-header { h3.section-header {
text-align: center; text-align: center;
@@ -328,6 +391,7 @@ h3.section-header {
.card { .card {
display: grid; display: grid;
grid-template-rows: 1fr auto; grid-template-rows: 1fr auto;
padding: 1px;
} }
.card_info { .card_info {
@@ -335,6 +399,15 @@ h3.section-header {
padding: 0.5em; padding: 0.5em;
} }
.changed-filters {
background-color: #111;
padding: 0.5em;
&[data-active='true'] {
color: lightgreen;
}
}
.card_controls { .card_controls {
display: flex; display: flex;
gap: 0.5em; gap: 0.5em;
@@ -358,33 +431,11 @@ 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;
} }
.card_regions {
display: flex;
justify-content: center;
label > input {
display: none;
}
label > span {
padding: 0.25em 0.5em;
margin: 0 0.25em;
cursor: pointer;
background-color: gray;
&.checked {
background-color: seagreen;
}
}
}
.card_timestamp { .card_timestamp {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
@@ -401,7 +452,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 {
@@ -410,8 +461,12 @@ h3.section-header {
} }
} }
.card_authors-search { .card_authors-search,
.card_sceneries-search {
margin: 1em 0; margin: 1em 0;
display: flex;
flex-direction: column;
align-items: center;
form { form {
display: flex; display: flex;
@@ -430,24 +485,63 @@ h3.section-header {
} }
} }
.card_actions { .section-filters {
width: 100%; display: grid;
padding: 0.5em; grid-template-columns: repeat(3, 1fr);
gap: 0.5em;
margin: 1em 0;
}
.filter-option { .section-filters > label {
max-width: 50%; position: relative;
margin: 0 auto; user-select: none;
-webkit-user-select: none;
-moz-user-select: none;
span {
cursor: pointer;
display: inline-block;
width: 100%;
text-align: center;
padding: 0.25em;
font-weight: bold;
background-color: forestgreen;
} }
span:hover {
background-color: #22aa22;
}
input[type='checkbox'] {
cursor: pointer;
position: absolute;
opacity: 0;
&:checked + span {
background-color: #444;
&:hover {
background-color: #555;
}
}
&:focus-visible + span {
outline: 1px solid var(--clr-primary);
}
}
}
.card_actions {
padding: 0.5em;
.action-buttons { .action-buttons {
display: flex; display: flex;
gap: 0.5em; gap: 0.5em;
width: 100%;
margin-top: 0.5em; margin-top: 0.5em;
button { button {
width: 50%; width: 100%;
margin: 0 auto; margin: 0 auto;
padding: 0.5em; padding: 0.5em;
@@ -471,33 +565,18 @@ h3.section-header {
} }
} }
.section-inputs {
display: grid;
grid-template-columns: repeat(3, 1fr);
gap: 0.5em;
margin: 1em 0;
}
.quick-actions div {
display: flex;
margin: 1em 0;
gap: 1em;
}
.slider { .slider {
display: flex; display: grid;
grid-template-columns: 1fr 50px 1fr;
align-items: center; align-items: center;
gap: 0.25em;
margin-bottom: 1em; margin-bottom: 1em;
&-value { &-value {
color: $accentCol; color: var(--clr-primary);
margin-right: 0.5em;
padding: 0.1em 0.2em; padding: 0.1em 0.2em;
} text-align: center;
&-content {
flex-grow: 2;
} }
&-input { &-input {
@@ -508,7 +587,6 @@ h3.section-header {
outline: none; outline: none;
min-width: 25%; min-width: 25%;
max-width: 120px;
&:focus-visible ~ * { &:focus-visible ~ * {
color: gold; color: gold;
@@ -525,13 +603,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);
} }
} }
@@ -542,14 +621,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);
} }
} }
@@ -578,4 +657,20 @@ h3.section-header {
} }
} }
} }
@include responsive.smallScreen{
.slider {
display: flex;
flex-wrap: wrap;
justify-content: center;
&-input {
width: 90%;
}
&-content {
text-align: center;
}
}
}
</style> </style>
@@ -0,0 +1,286 @@
<template>
<div
class="dropdown"
@keydown.esc="showDropdown = false"
v-click-outside="() => (showDropdown = false)"
>
<div class="bg" v-if="showDropdown" @click="showDropdown = false"></div>
<button class="filter-button btn--filled btn--image" @click="toggleDropdown" ref="button">
<img src="/images/icon-stats.svg" alt="Open filters icon" />
{{ $t('station-stats.stats-button') }}
</button>
<transition name="dropdown-anim">
<div class="dropdown_wrapper" v-if="showDropdown">
<div>
<h1 class="stats-title text--primary">
<img src="/images/icon-stats.svg" alt="Open filters icon" />
{{ $t('station-stats.title') }}
</h1>
<hr style="margin: 0.5em 0" />
<div v-if="uFactor > -1 || avgTimetableCount > -1 || trackCount.all > 0">
<div class="badges-container">
<div class="badge stat-badge">
<span>
{{ $t('station-stats.u-factor') }}
<a
href="https://td2.info.pl/dyskusje/wspolczynnik-ugla-czy-to-ma-sens/msg81011/#msg81011"
target="_blank"
:data-tooltip="$t('station-stats.u-factor-tooltip')"
>(?)</a
>:
</span>
<span>
<b class="u-factor" :style="calculateFactorStyle()">
{{ uFactor >= 0 ? uFactor.toFixed(2) : '---' }}
</b>
</span>
</div>
<div class="badge stat-badge">
<span>{{ $t('station-stats.avg-timetable-count') }}</span>
<span>
<b>{{ avgTimetableCount >= 0 ? avgTimetableCount.toFixed(2) : '---' }}</b>
</span>
</div>
</div>
<hr style="margin: 0.5em 0" />
<div class="badges-container">
<div class="badge stat-badge">
<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>)
</span>
</div>
</div>
<hr style="margin: 0.5em 0" />
<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 tabindex="0" @focus="() => (showDropdown = false)"></div>
</div>
</transition>
</div>
</template>
<script lang="ts">
import { defineComponent } from 'vue';
import { useMainStore } from '../../store/mainStore';
export default defineComponent({
data() {
return {
mainStore: useMainStore(),
showDropdown: false
};
},
methods: {
toggleDropdown() {
this.showDropdown = !this.showDropdown;
},
calculateFactorStyle() {
if (this.uFactor <= 0) return '';
const norm = Math.max(Math.min(this.uFactor / 2, 1), 0);
const lerp = 120 * norm;
return `color: hsl(${lerp}, 100%, 60%)`;
}
},
computed: {
uFactor() {
const activeDispatchers = this.mainStore.activeSceneryList.filter(
(scenery) => scenery.region == this.mainStore.region.id && scenery.dispatcherId != -1
);
const activeTrains = this.mainStore.trainList.filter(
(train) => train.region == this.mainStore.region.id
);
return activeDispatchers.length != 0 ? activeTrains.length / activeDispatchers.length : -1;
},
avgTimetableCount() {
const regionSceneries = this.mainStore.activeSceneryList.filter((sc) => {
return sc.region == this.mainStore.region.id;
});
const timetableCountSum = regionSceneries.reduce((acc, sc) => {
acc += sc.scheduledTrainCount.all;
return acc;
}, 0);
if (regionSceneries.length == 0) return -1;
return timetableCountSum / regionSceneries.length;
},
trackCount() {
return this.mainStore.allStationInfo
.filter(
(st) =>
st.onlineInfo?.dispatcherId != -1 &&
st.onlineInfo?.region == this.mainStore.region.id &&
st.generalInfo?.routes
)
.reduce(
(acc, st) => {
const { routes } = st.generalInfo!;
acc.all++;
if (
routes.single.filter((r) => !r.isInternal).length > 0 &&
routes.double.filter((r) => !r.isInternal).length > 0
) {
acc.crossTrack++;
if (
routes.single.some((r) => r.isElectric) &&
routes.double.some((r) => r.isElectric)
)
acc.crossTrackElectric++;
}
[...routes.single, ...routes.double].forEach((r) => {
if (r.isInternal) return;
acc[r.routeTracks == 2 ? 'twoWay' : 'oneWay'] += 1;
if (r.isElectric) acc[r.routeTracks == 2 ? 'twoWayElectric' : 'oneWayElectric'] += 1;
});
return acc;
},
{
oneWay: 0,
oneWayElectric: 0,
twoWay: 0,
twoWayElectric: 0,
crossTrack: 0,
crossTrackElectric: 0,
all: 0
}
);
},
spawnCount() {
return this.mainStore.activeSceneryList.reduce(
(acc, scenery) => {
if (scenery.region != this.mainStore.region.id) return acc;
scenery.spawns.forEach((spawn) => {
if (/EZT|POS|OSOB/i.test(spawn.spawnName)) acc['passenger'] += 1;
if (/TOW/i.test(spawn.spawnName)) acc['freight'] += 1;
if (/LUZ|SM/i.test(spawn.spawnName)) acc['loco'] += 1;
if (/ALL/i.test(spawn.spawnName)) acc['all'] += 1;
});
return acc;
},
{ passenger: 0, freight: 0, loco: 0, all: 0 }
);
}
}
});
</script>
<style lang="scss" scoped>
@use '../../styles/dropdown';
@use '../../styles/badge';
@use '../../styles/responsive';
h1.stats-title img {
vertical-align: text-bottom;
}
.badges-container {
display: flex;
flex-wrap: wrap;
gap: 0.5em;
}
.u-factor {
[data-factor-low='true'] {
color: #ddd;
}
[data-factor-mediocre='true'] {
color: lightgreen;
}
[data-factor-high='true'] {
color: greenyellow;
}
[data-factor-highest='true'] {
color: rgb(22, 245, 22);
}
}
.no-data {
font-size: 1.1em;
color: #ccc;
}
@include responsive.smallScreen {
h1.stats-title {
text-align: center;
}
}
</style>
+358 -350
View File
@@ -1,371 +1,383 @@
<template> <template>
<section class="station_table"> <section class="station_table">
<transition name="status-anim" mode="out-in"> <Loading
<div class="table_wrapper" :key="apiStore.dataStatuses.connection"> v-if="apiStore.dataStatuses.connection == Status.Loading && filteredStationList.length == 0"
<table> />
<thead>
<tr> <table v-else-if="filteredStationList.length > 0">
<th <thead>
v-for="headerName in headIds" <tr>
:key="headerName" <th
@click="changeSorter(headerName)" v-for="headerName in headIds"
class="header-text" :key="headerName"
:class="headerName" @click="changeSorter(headerName)"
class="header-text"
:class="headerName"
>
<span class="header_wrapper">
<div v-html="$t(`sceneries.headers.${headerName}`)"></div>
<img
class="sort-icon"
v-if="activeSorter.headerName == headerName"
:src="`/images/icon-arrow-${activeSorter.dir == 1 ? 'asc' : 'desc'}.svg`"
alt="sort icon"
/>
</span>
</th>
<th
v-for="headerName in headIconsIds"
:key="headerName"
@click="changeSorter(headerName)"
class="header-image"
:class="headerName"
>
<span class="header_wrapper">
<img
:src="`/images/icon-${headerName}.svg`"
:alt="headerName"
:title="$t(`sceneries.headers.${headerName}`)"
/>
<img
class="sort-icon"
v-if="activeSorter.headerName == headerName"
:src="`/images/icon-arrow-${activeSorter.dir == 1 ? 'asc' : 'desc'}.svg`"
alt="sort icon"
/>
</span>
</th>
</tr>
</thead>
<tbody>
<router-link
v-for="station in filteredStationList"
class="a-row"
role="row"
:key="station.name"
@click.right.prevent="openForumSite($event, station.generalInfo?.url)"
@keydown.space.prevent="openForumSite($event, station.generalInfo?.url)"
:to="getSceneryRoute(station)"
>
<td class="station-name" :class="station.generalInfo?.availability">
<b v-if="station.generalInfo?.project" style="color: salmon">{{
station.generalInfo.project
}}</b>
{{ station.name }}
</td>
<td class="station-level">
<span v-if="station.generalInfo">
<span
v-if="
station.generalInfo.reqLevel > -1 &&
station.generalInfo.availability != 'nonPublic' &&
station.generalInfo.availability != 'unavailable'
"
:style="calculateExpStyle(station.generalInfo.reqLevel)"
> >
<span class="header_wrapper"> {{ station.generalInfo.reqLevel >= 2 ? station.generalInfo.reqLevel : 'L' }}
<div v-html="$t(`sceneries.headers.${headerName}`)"></div> </span>
<img <span v-else-if="station.generalInfo.availability == 'abandoned'">
class="sort-icon" <img
v-if="sorterActive.headerName == headerName" src="/images/icon-abandoned.svg"
:src="`/images/icon-arrow-${sorterActive.dir == 1 ? 'asc' : 'desc'}.svg`" alt="non-public"
alt="sort icon" :title="$t('sceneries.info.abandoned')"
/> />
</span> </span>
</th>
<th <span v-else-if="station.generalInfo.availability == 'nonPublic'">
v-for="headerName in headIconsIds" <img
:key="headerName" src="/images/icon-lock.svg"
@click="changeSorter(headerName)" alt="non-public"
class="header-image" :title="$t('sceneries.info.non-public')"
:class="headerName" />
</span>
<span v-else>
<img
src="/images/icon-unavailable.svg"
alt="unavailable"
:title="$t('sceneries.info.unavailable')"
/>
</span>
</span>
<span v-else> ? </span>
</td>
<td class="station-status">
<StationStatusBadge
:isOnline="station.onlineInfo ? true : false"
:dispatcherStatus="station.onlineInfo?.dispatcherStatus"
/>
</td>
<td class="station-dispatcher-name">
<span v-if="station.onlineInfo?.dispatcherName">
<b
v-if="apiStore.donatorsData.includes(station.onlineInfo.dispatcherName)"
@click.prevent="openDonationCard"
data-tooltip-type="DonatorTooltip"
:data-tooltip-content="$t('donations.dispatcher-message')"
> >
<span class="header_wrapper"> <img src="/images/icon-diamond.svg" alt="" />
<img {{ station.onlineInfo.dispatcherName }}
:src="`/images/icon-${headerName}.svg`" </b>
:alt="headerName"
:title="$t(`sceneries.headers.${headerName}`)"
/>
<img <div v-else>
class="sort-icon" {{ station.onlineInfo.dispatcherName }}
v-if="sorterActive.headerName == headerName" </div>
:src="`/images/icon-arrow-${sorterActive.dir == 1 ? 'asc' : 'desc'}.svg`" </span>
alt="sort icon" </td>
/>
</span>
</th>
</tr>
</thead>
<tbody> <td class="station-dispatcher-exp">
<tr <span
v-for="station in stations" v-if="station.onlineInfo && station.onlineInfo?.dispatcherExp != -1"
:class="{ 'last-selected': lastSelectedStationName == station.name }" :style="
:key="station.name" calculateExpStyle(
@click.left="setScenery(station.name)" station.onlineInfo.dispatcherExp,
@click.right="openForumSite($event, station.generalInfo?.url)" station.onlineInfo.dispatcherIsSupporter
@keydown.enter="setScenery(station.name)" )
@keydown.space="openForumSite($event, station.generalInfo?.url)" "
tabindex="0"
> >
<td class="station-name" :class="station.generalInfo?.availability"> {{ station.onlineInfo.dispatcherExp < 2 ? 'L' : station.onlineInfo.dispatcherExp }}
<b v-if="station.generalInfo?.project" style="color: salmon">{{ </span>
station.generalInfo.project </td>
}}</b>
{{ station.name }}
</td>
<td class="station-level"> <td class="station-tracks">
<span v-if="station.generalInfo"> <div v-if="station.generalInfo">
<span <span
v-if=" v-if="station.generalInfo.routes.singleElectrifiedNames.length != 0"
station.generalInfo.reqLevel > -1 && class="track catenary"
station.generalInfo.availability != 'nonPublic' && :title="`${$t('sceneries.info.single-track-routes-catenary')}${
station.generalInfo.availability != 'unavailable' station.generalInfo.routes.singleElectrifiedNames.length
" }`"
:style="calculateExpStyle(station.generalInfo.reqLevel)"
>
{{ station.generalInfo.reqLevel >= 2 ? station.generalInfo.reqLevel : 'L' }}
</span>
<span v-else-if="station.generalInfo.availability == 'abandoned'">
<img
src="/images/icon-abandoned.svg"
alt="non-public"
:title="$t('sceneries.info.abandoned')"
/>
</span>
<span v-else-if="station.generalInfo.availability == 'nonPublic'">
<img
src="/images/icon-lock.svg"
alt="non-public"
:title="$t('sceneries.info.non-public')"
/>
</span>
<span v-else>
<img
src="/images/icon-unavailable.svg"
alt="unavailable"
:title="$t('sceneries.info.unavailable')"
/>
</span>
</span>
<span v-else> ? </span>
</td>
<td class="station-status">
<StationStatusBadge
:isOnline="station.onlineInfo ? true : false"
:dispatcherStatus="station.onlineInfo?.dispatcherStatus"
/>
</td>
<td class="station-dispatcher-name">
<span v-if="station.onlineInfo?.dispatcherName">
<b
v-if="apiStore.donatorsData.includes(station.onlineInfo.dispatcherName)"
:title="$t('donations.dispatcher-message')"
@click.stop="openDonationModal"
>
<img src="/images/icon-diamond.svg" alt="" />
{{ station.onlineInfo.dispatcherName }}
</b>
<div v-else>
{{ station.onlineInfo.dispatcherName }}
</div>
</span>
</td>
<td class="station-dispatcher-exp">
<span
v-if="station.onlineInfo && station.onlineInfo?.dispatcherExp != -1"
:style="
calculateExpStyle(
station.onlineInfo.dispatcherExp,
station.onlineInfo.dispatcherIsSupporter
)
"
>
{{
station.onlineInfo.dispatcherExp < 2 ? 'L' : station.onlineInfo.dispatcherExp
}}
</span>
</td>
<td class="station-tracks">
<div v-if="station.generalInfo">
<span
v-if="station.generalInfo.routes.singleElectrifiedNames.length != 0"
class="track catenary"
:title="`${$t('sceneries.info.single-track-routes-catenary')}${
station.generalInfo.routes.singleElectrifiedNames.length
}`"
>
{{ station.generalInfo.routes.singleElectrifiedNames.length }}
</span>
<span
v-if="station.generalInfo.routes.singleOtherNames.length != 0"
class="track no-catenary"
:title="`${$t('sceneries.info.single-track-routes-other')}${
station.generalInfo.routes.singleOtherNames.length
}`"
>
{{ station.generalInfo.routes.singleOtherNames.length }}
</span>
</div>
</td>
<td class="station-tracks">
<div v-if="station.generalInfo">
<span
v-if="station.generalInfo.routes.doubleElectrifiedNames.length != 0"
class="track catenary"
:title="`${$t('sceneries.info.double-track-routes-catenary')}${
station.generalInfo.routes.doubleElectrifiedNames.length
}`"
>
{{ station.generalInfo.routes.doubleElectrifiedNames.length }}
</span>
<span
v-if="station.generalInfo.routes.doubleOtherNames.length != 0"
class="track no-catenary"
:title="`${$t('sceneries.info.double-track-routes-other')}${
station.generalInfo.routes.doubleOtherNames.length
}`"
>
{{ station.generalInfo.routes.doubleOtherNames.length }}
</span>
</div>
</td>
<td class="station-info">
<span
v-if="station.generalInfo?.signalType"
class="scenery-icon icon-info"
:class="station.generalInfo?.controlType.replace('+', '-')"
:title="
$t('sceneries.info.control-type') +
$t(`controls.${station.generalInfo?.controlType}`)
"
v-html="getControlTypeAbbrev(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?.SUP"
class="icon-info"
src="/images/icon-SUP.svg"
alt="SUP (RASP-UZK)"
:title="$t('sceneries.info.SUP')"
/>
<img
v-if="station.generalInfo?.ASDEK"
class="icon-info"
src="/images/icon-ASDEK.svg"
alt="dSAT ASDEK"
:title="$t('sceneries.info.ASDEK')"
/>
<img
v-if="!station.generalInfo"
class="icon-info"
src="/images/icon-unknown.svg"
alt="icon-unknown"
:title="$t('sceneries.info.unknown')"
/>
</td>
<td class="station-users" :class="{ inactive: !station.onlineInfo }">
<span class="text--primary">{{ station.onlineInfo?.currentUsers ?? '-' }}</span>
/
<span class="text--primary">{{ station.onlineInfo?.maxUsers ?? '-' }}</span>
</td>
<td class="station-likes" :class="{ inactive: !station.onlineInfo }">
<span>{{ station.onlineInfo?.dispatcherRate ?? '-' }}</span>
</td>
<td class="station-spawns" :class="{ inactive: !station.onlineInfo }">
<span>{{ station.onlineInfo?.spawns.length ?? '-' }}</span>
</td>
<td
class="station-schedules all"
style="width: 30px"
:class="{ inactive: !station.onlineInfo }"
> >
{{ station.onlineInfo?.scheduledTrainCount.all ?? '-' }} {{ station.generalInfo.routes.singleElectrifiedNames.length }}
</td> </span>
<td <span
class="station-schedules unconfirmed" v-if="station.generalInfo.routes.singleOtherNames.length != 0"
style="width: 30px" class="track no-catenary"
:class="{ inactive: !station.onlineInfo }" :title="`${$t('sceneries.info.single-track-routes-other')}${
station.generalInfo.routes.singleOtherNames.length
}`"
> >
{{ station.onlineInfo?.scheduledTrainCount.unconfirmed ?? '-' }} {{ station.generalInfo.routes.singleOtherNames.length }}
</td> </span>
</div>
</td>
<td <td class="station-tracks">
class="station-schedules confirmed" <div v-if="station.generalInfo">
style="width: 30px" <span
:class="{ inactive: !station.onlineInfo }" v-if="station.generalInfo.routes.doubleElectrifiedNames.length != 0"
class="track catenary"
:title="`${$t('sceneries.info.double-track-routes-catenary')}${
station.generalInfo.routes.doubleElectrifiedNames.length
}`"
> >
{{ station.onlineInfo?.scheduledTrainCount.confirmed ?? '-' }} {{ station.generalInfo.routes.doubleElectrifiedNames.length }}
</td> </span>
</tr>
</tbody>
</table>
<Loading v-if="apiStore.dataStatuses.connection == Status.Loading" /> <span
v-if="station.generalInfo.routes.doubleOtherNames.length != 0"
class="track no-catenary"
:title="`${$t('sceneries.info.double-track-routes-other')}${
station.generalInfo.routes.doubleOtherNames.length
}`"
>
{{ station.generalInfo.routes.doubleOtherNames.length }}
</span>
</div>
</td>
<div class="no-stations" v-else-if="stations.length == 0"> <td class="station-info">
{{ $t('sceneries.no-stations') }} <span
</div> v-if="station.generalInfo?.signalType"
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?.SUP"
class="icon-info"
src="/images/icon-SUP.svg"
alt="SUP (RASP-UZK)"
:title="$t('sceneries.info.SUP')"
/>
<img
v-if="station.generalInfo?.ASDEK"
class="icon-info"
src="/images/icon-ASDEK.svg"
alt="dSAT ASDEK"
:title="$t('sceneries.info.ASDEK')"
/>
<img
v-if="!station.generalInfo"
class="icon-info"
src="/images/icon-unknown.svg"
alt="icon-unknown"
:title="$t('sceneries.info.unknown')"
/>
</td>
<td
class="station-users"
:class="{ inactive: !station.onlineInfo }"
data-tooltip-type="UsersTooltip"
:data-tooltip-content="JSON.stringify(station.onlineInfo?.stationTrains ?? [])"
>
<span class="text--primary">{{
station.onlineInfo?.stationTrains?.length ?? '-'
}}</span>
/
<span class="text--primary">{{ station.onlineInfo?.maxUsers ?? '-' }}</span>
</td>
<td class="station-likes" :class="{ inactive: !station.onlineInfo }">
<span>{{ station.onlineInfo?.dispatcherRate ?? '-' }}</span>
</td>
<td
class="station-spawns"
:class="{ inactive: !station.onlineInfo }"
data-tooltip-type="SpawnsTooltip"
:data-tooltip-content="JSON.stringify(station.onlineInfo?.spawns ?? [])"
>
<span>{{ station.onlineInfo?.spawns.length ?? '-' }}</span>
</td>
<td
class="station-schedules all"
style="width: 30px"
:class="{ inactive: !station.onlineInfo }"
>
{{ station.onlineInfo?.scheduledTrainCount.all ?? '-' }}
</td>
<td
class="station-schedules unconfirmed"
style="width: 30px"
:class="{ inactive: !station.onlineInfo }"
>
{{ station.onlineInfo?.scheduledTrainCount.unconfirmed ?? '-' }}
</td>
<td
class="station-schedules confirmed"
style="width: 30px"
:class="{ inactive: !station.onlineInfo }"
>
{{ station.onlineInfo?.scheduledTrainCount.confirmed ?? '-' }}
</td>
</router-link>
</tbody>
</table>
<div class="no-stations" v-else>
<div>
{{ $t('sceneries.no-stations') }} (region: <b>{{ mainStore.region.name }}</b
>)
</div> </div>
</transition>
<div class="text--primary" v-if="getChangedFilters(filters).length != 0">
{{ $t('sceneries.active-filters') }}
</div>
</div>
</section> </section>
</template> </template>
<script lang="ts"> <script lang="ts">
import { defineComponent, PropType } from 'vue'; import { defineComponent, inject, computed } from 'vue';
import dateMixin from '../../mixins/dateMixin';
import stationInfoMixin from '../../mixins/stationInfoMixin';
import styleMixin from '../../mixins/styleMixin';
import Station from '../../scripts/interfaces/Station';
import { useStationFiltersStore } from '../../store/stationFiltersStore';
import { useMainStore } from '../../store/mainStore';
import Loading from '../Global/Loading.vue';
import { HeadIdsTypes, headIconsIds, headIds } from '../../scripts/data/stationHeaderNames';
import StationStatusBadge from '../Global/StationStatusBadge.vue'; import StationStatusBadge from '../Global/StationStatusBadge.vue';
import { Status } from '../../typings/common'; import Loading from '../Global/Loading.vue';
import dateMixin from '../../mixins/dateMixin';
import styleMixin from '../../mixins/styleMixin';
import { useApiStore } from '../../store/apiStore'; import { useApiStore } from '../../store/apiStore';
import { useMainStore } from '../../store/mainStore';
import { Station, Status } from '../../typings/common';
import { useTooltipStore } from '../../store/tooltipStore';
import { getChangedFilters } from '../../managers/stationFilterManager';
import { ActiveSorter, HeadIdsType, headIconsIds, headIds } from './typings';
import { filterStations, sortStations } from './utils';
export default defineComponent({ export default defineComponent({
props: { emits: ['toggleDonationCard'],
stations: {
type: Array as PropType<Station[]>,
required: true
}
},
emits: ['toggleDonationModal'],
components: { Loading, StationStatusBadge }, components: { Loading, StationStatusBadge },
mixins: [styleMixin, dateMixin, stationInfoMixin], mixins: [styleMixin, dateMixin],
data: () => ({ data: () => ({
headIconsIds, headIconsIds,
headIds, headIds,
lastSelectedStationName: '' getChangedFilters
}), }),
computed: {
sorterActive() {
return this.stationFiltersStore.sorterActive;
}
},
setup() { setup() {
const mainStore = useMainStore(); const mainStore = useMainStore();
const apiStore = useApiStore(); const apiStore = useApiStore();
const stationFiltersStore = useStationFiltersStore(); const tooltipStore = useTooltipStore();
const filters = inject('StationsView_filters') as Record<string, any>;
const activeSorter = inject('StationsView_activeSorter') as ActiveSorter;
const filteredStationList = computed(() =>
mainStore.allStationInfo
.filter((station) => filterStations(station, filters))
.sort((a, b) => sortStations(a, b, activeSorter))
);
return { return {
Status: Status.Data, Status: Status.Data,
stationFiltersStore,
mainStore, mainStore,
apiStore apiStore,
tooltipStore,
filters,
filteredStationList,
activeSorter
}; };
}, },
methods: { methods: {
setScenery(name: string) { getSceneryRoute(station: Station) {
const station = this.stations.find((station) => station.name === name); // TODO: Hide tooltips when navigating away
if (!station) return;
this.lastSelectedStationName = station.name; return {
this.$router.push({
name: 'SceneryView', name: 'SceneryView',
query: { query: {
station: station.name.replaceAll(' ', '_'), station: station.name,
region: this.$route.query.region || undefined region: this.$route.query.region || undefined
} }
}); };
}, },
openDonationModal(e: Event) { openDonationCard(e: Event) {
this.$emit('toggleDonationModal', true); this.$emit('toggleDonationCard', true);
this.mainStore.modalLastClickedTarget = e.target; this.mainStore.modalLastClickedTarget = e.target;
this.tooltipStore.hide();
}, },
openForumSite(e: Event, url: string | undefined) { openForumSite(e: Event, url: string | undefined) {
@@ -374,49 +386,41 @@ export default defineComponent({
window.open(url, '_blank'); window.open(url, '_blank');
}, },
changeSorter(headerName: HeadIdsTypes) { changeSorter(headerName: HeadIdsType) {
if (headerName == 'general') return; if (headerName == 'general') return;
this.stationFiltersStore.changeSorter(headerName); if (headerName == this.activeSorter.headerName)
this.activeSorter.dir = -1 * this.activeSorter.dir;
else this.activeSorter.dir = 1;
this.activeSorter.headerName = headerName;
} }
} }
}); });
</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;
.change-anim { .station_table {
&-enter-active, height: calc(100vh - 11em);
&-leave-active { max-height: 2000px;
transition: opacity 100ms ease-in; min-height: 500px;
}
&-enter,
&-leave-to {
opacity: 0;
}
}
.table_wrapper {
overflow: auto; overflow: auto;
font-weight: 500; font-weight: 500;
height: 90vh;
min-height: 550px;
} }
.no-stations { .no-stations {
text-align: center; text-align: center;
font-size: 1.5em; font-size: 1.25em;
padding: 1em; padding: 1em;
margin: 1em 0; background: #1a1a1a;
line-height: 1.5em;
background: #333;
} }
table { table {
@@ -426,14 +430,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;
} }
@@ -472,7 +478,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;
@@ -493,17 +499,19 @@ table {
} }
} }
tr { tr,
.a-row {
background-color: $rowCol; background-color: $rowCol;
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 {
@@ -517,7 +525,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;
@@ -530,7 +538,7 @@ tr {
max-width: 200px; max-width: 200px;
&.default { &.default {
color: $accentCol; color: var(--clr-primary);
} }
&.nonPublic { &.nonPublic {

Some files were not shown because too many files have changed in this diff Show More