74 Commits

Author SHA1 Message Date
ce671f4b10 🐛 Enable virtual env before running poetry
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-14 23:28:49 +01:00
94d87c1c0e 🐛 Fix invalid running directory (part 2)
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-14 23:24:16 +01:00
71f7519b2f 🐛 Fix invalid running directory
Some checks failed
ci/woodpecker/pr/lint Pipeline failed
2024-02-14 23:10:26 +01:00
99aeaae9f2 🚧 Skip the PATH update
Some checks failed
ci/woodpecker/pr/lint Pipeline failed
2024-02-14 23:06:43 +01:00
06d446c3f9 🚧 Edit PATH before loading python virtual env
Some checks failed
ci/woodpecker/pr/lint Pipeline failed
2024-02-14 22:58:04 +01:00
001c7330c6 🎨 Prepare linters environment once
Some checks failed
ci/woodpecker/pr/lint Pipeline failed
2024-02-14 22:49:55 +01:00
d4d24d13d2 🐛 Install poetry for each step
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-13 23:39:46 +01:00
f1cafc7484 🚧 Add dummy commit
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-13 23:05:45 +01:00
30020d17cc 🐛 Fix typo 2024-02-13 23:02:40 +01:00
85bdf2656c 🚧 Comment temporary the backend linter when conditions 2024-02-13 22:56:45 +01:00
fe67cd0d98 🚧 Dummy commit 2024-02-13 22:36:34 +01:00
221125198d 🐛 Image shall be set for each CI step 2024-02-13 22:09:09 +01:00
fe05e2a2c5 🎨 Factorize backend linters steps 2024-02-13 20:57:18 +01:00
f964ba9027 🚧 Add dummy commit to check the reviewdog filter mode
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-12 23:38:34 +01:00
117d549ea7 🚧 Redirect reviewdog stderr to stdout
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 21:38:46 +01:00
529315bf50 🚧 Disable reviewdog filter and create empty .reviewdog.yml file
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 21:31:23 +01:00
35882bda31 🔊 Increase reviewdog logs
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 21:04:22 +01:00
688ffeb7f8 🚧 Add the sqlalchemy dep to the linters poetry group
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 20:14:44 +01:00
0993c42557 🚧 Poetry shell -> poetry run
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 20:02:47 +01:00
2b0eb40ab1 🚧 Run linters in poetry virtual environment
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 19:59:10 +01:00
8082f7f012 🚧 Remove the deps installation and disable mypy caching
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 19:56:18 +01:00
2d9d7c51b5 🚧 Add gcc to the CI image
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 19:52:34 +01:00
63cdbcb46e 🚧 Install dependencies before running the linters
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 19:46:12 +01:00
825a852c67 🚧 Fix reviewdog path
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 19:34:51 +01:00
812d7c0a61 🚧 Format reviewdog command and use temp file
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 19:31:47 +01:00
63abb677cb 🚧 Try to run linters on pull_request event
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 19:13:13 +01:00
a089bbbf78 🔊 Print env variables
All checks were successful
ci/woodpecker/push/lint Pipeline was successful
2024-02-11 19:09:06 +01:00
55401dcd0c 🚧 Add gitea_address secret
All checks were successful
ci/woodpecker/push/lint Pipeline was successful
2024-02-11 19:05:15 +01:00
58c18a0479 🚧 Enable REVIEWDOG_GITEA_API_TOKEN
All checks were successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 18:54:05 +01:00
2eb8d59a0e 🚧 Fix typo
All checks were successful
ci/woodpecker/pr/lint Pipeline was successful
ci/woodpecker/push/lint Pipeline was successful
2024-02-11 18:38:47 +01:00
137c28387c 🚧 Fix eviewdog path
All checks were successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 18:37:23 +01:00
0cddf9dc7e 🚧 Fix reviewdog version and setup ruff
All checks were successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 18:34:51 +01:00
bccbfe0ff3 🚧 First reviewdog try
All checks were successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 18:30:43 +01:00
10ecd40ef5 💚 Force the linters to run even if the previous one fails.
All checks were successful
ci/woodpecker/push/lint Pipeline was successful
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 17:52:28 +01:00
6687c2de15 💚 Run backend linters on commits only
Some checks failed
ci/woodpecker/push/lint Pipeline failed
ci/woodpecker/pr/lint Pipeline failed
Try to force the linters to run even if the previous one fails.
2024-02-11 17:48:01 +01:00
c8732a0375 Run Mypy linter on backend pull requests
Some checks failed
ci/woodpecker/push/lint Pipeline failed
ci/woodpecker/pr/lint Pipeline failed
2024-02-11 17:40:06 +01:00
ff808e7c8f 💚 Don't run linters on master, develop and release/* branches
Some checks failed
ci/woodpecker/push/lint Pipeline failed
ci/woodpecker/pr/lint Pipeline was successful
2024-02-11 17:24:24 +01:00
7ac719db24 👷 Run Ruff linter on back-end pull requests
All checks were successful
ci/woodpecker/manual/lint Pipeline was successful
2024-02-11 17:06:04 +01:00
a8277fd018 👷 First try
All checks were successful
ci/woodpecker/push/woodpecker Pipeline was successful
ci/woodpecker/manual/woodpecker Pipeline was successful
ci/woodpecker/pr/woodpecker Pipeline was successful
2024-02-03 18:04:09 +01:00
e98b24303c 🎨 Cleanup code formatting 2024-02-02 23:02:39 +01:00
f6bd241db3 Merge branch 'readme' into develop 2024-01-08 23:05:38 +01:00
eead35f822 📝 Add Docs section to the TODO list 2024-01-08 23:03:59 +01:00
3ff9a91bd0 Merge branch 'readme' into develop 2024-01-07 11:44:23 +01:00
d7bef20ffc 📄 Rewrite reference of the origin of the project name 2024-01-07 11:42:44 +01:00
c2c8c81759 📄 Replace WIP tag with a reference to the origin of the project name 2024-01-07 11:39:04 +01:00
98dbf5dfe2 📄 Add WIP tag to the README file 2024-01-07 11:35:46 +01:00
3873fdb719 Merge branch 'readme' into develop 2024-01-07 11:30:03 +01:00
4236f33cb9 📝 Add technical stack description to the README 2024-01-07 11:19:14 +01:00
60ce90b633 🚧 Add C4 Container diagram 2024-01-06 17:45:58 +01:00
c52fc69560 🚧 Add plantuml block to the README to test gitea rendering 2024-01-06 13:32:17 +01:00
e9a13c662e 📝 Change the presentation video chroma subsampling
Cf. https://bugzilla.mozilla.org/show_bug.cgi?id=1368063
2023-12-22 11:36:01 +01:00
93625f12b1 📝 Add thumbnail to the presentation video 2023-12-22 11:27:30 +01:00
b7ed3f83b8 📝 Add main readme file
Signed-off-by: Adrien <me@adrien.run>
2023-12-22 11:03:29 +01:00
f1a47c9621 📝 First try to embed a video in the readme page 2023-12-22 10:59:44 +01:00
7843309f0a 📝 Add a video showing what carrramba-encore-rate service is 2023-12-22 10:48:22 +01:00
cebc9077c3 👽️ Add of the StopAreaStopAssociationFields pdeid and pdeversion fields
These fields have been added by IDFM in its relations resource.
For now, these fields are not used.
2023-10-22 23:41:27 +02:00
f862e124a6 Merge branch 'k8s-integration' into develop 2023-10-22 23:37:44 +02:00
1bb75b28eb ♻️ Use of relative imports for api modules 2023-10-22 23:34:58 +02:00
0a7337a313 ♻️ Put api_server and db_updater scripts on the backend root 2023-10-22 23:31:35 +02:00
fcd1ae78c3 🐛 Fix IP to bind in the api config file for local use 2023-09-20 22:24:02 +02:00
ec1b4100a3 🐛 Fix validation issue in /api/stop/{stop_id}/destinations responses 2023-09-20 22:22:13 +02:00
37ec05bf3b Merge branch 'k8s-integration' into develop 2023-09-20 22:14:56 +02:00
3434802b31 🎨 Reorganize back-end code 2023-09-20 22:08:32 +02:00
bdbc72ab39 🐛 Front: Fix URL used to fetch transport mode representation 2023-09-10 12:25:38 +02:00
4cc8f60076 🐛 Front: Use the public API server to fetch data 2023-09-10 12:17:48 +02:00
cf5c4c6224 🔒️ Fix CORS allowed origins and methods 2023-09-10 12:07:20 +02:00
f69aee1c9c 🔒️ Remove driver and password from configuration file
Password will be provided by vault using an env variable.
2023-09-10 12:04:25 +02:00
8c493f8fab ♻️ Remove pg_trgm creation from the db session init
The pg_trgm extension will be created during db init, by the db-updated image.
2023-09-10 11:46:24 +02:00
4fce832db5 ♻️ Rename docker file building api image 2023-09-10 11:45:08 +02:00
bfc669cd11 ♻️ Use pydantic-settings to handle config file 2023-09-09 23:35:18 +02:00
4056b3a739 🐛 Error raised by frontend Map component if no stop found 2023-09-09 23:18:03 +02:00
f7f0fdb980 ️ Use of integer to store Line and Stop id
Update Line and Stop schemas.
2023-09-09 23:05:18 +02:00
6c149e844b 💥 Remove /widget static endpoint
This endpoint shall be served by a dedicated static HTTP server.
2023-06-13 05:45:33 +02:00
f5529bba24 Merge branch 'remove-db-filling-from-backend' into develop 2023-06-13 05:44:00 +02:00
62 changed files with 2281 additions and 2131 deletions

2
.gitattributes vendored Normal file
View File

@@ -0,0 +1,2 @@
medias/presentation.png filter=lfs diff=lfs merge=lfs -text
medias/presentation.mp4 filter=lfs diff=lfs merge=lfs -text

124
README.md Normal file
View File

@@ -0,0 +1,124 @@
# 🚧 Carrramba! Encore raté! 🚧
Resident of the Ile-de-France,
- Tired of missing your bus/train/metro ?
- Tired of having to walk to know where your bus/train/metro will stop ?
- Are you looking to display its next passages in the same way as Ile De France Mobilité ?
**Visit [carrramba.adrien.run](https://carrramba.adrien.run/)**
[![Presentation](medias/presentation.png)](medias/presentation.mp4)
# Technical stack
The following figure list the building blocks (docker images) and their interactions to provide the service:
```plantuml
@startuml
!include <C4/C4_Container>
!define ICONURL https://raw.githubusercontent.com/tupadr3/plantuml-icon-font-sprites/v2.4.0
!includeurl ICONURL/common.puml
!includeurl ICONURL/font-awesome-5/users.puml
!includeurl ICONURL/devicons2/redis.puml
!includeurl ICONURL/devicons2/sqlalchemy.puml
!includeurl ICONURL/devicons2/postgresql.puml
title Carrramba-encore-rate - Back-end
LAYOUT_WITH_LEGEND()
Person(user, "User", "The service user", $sprite="users")
System_Boundary(cb1, "carrramba-encore-rate") {
Container(app, "SPA", "SolidJS", "The graphical interface used by users to consume services provided")
Container(api, "carrramba-encore-rate-api", "FastAPI / Pydantic 2 / SQLAlchemy 2", "Provides the functionalities serving API endpoints", $sprite="sqlalchemy")
ContainerDb(db, "Postgres", "PostgreSQL database", "Stores stops/stop areas/lines/shapes and associations information.", $sprite="postgresql")
ContainerDb(cache, "In-Memory cache", "Redis", "Store previously computed results (stop, line, destination, next passages, shapes).", $sprite="redis")
Container(db_updater, "db-updater", "Sync the service static data with the IDFM ones")
}
System_Boundary(idfm, "IDFM") {
Container_Ext(static_, "Static data")
Container_Ext(dynamic_, "Dynamic data")
}
Rel(user, app, "Uses")
Rel(app, api, "Uses", "JSON/HTTPS")
Rel_R(api, db, "Reads from", "sync, PSQL")
Rel_L(api, cache, "Reads from and writes to", "sync, REdis Serialization Protocol")
Rel(api, dynamic_, "Get next passages", "JSON/HTTPS")
Rel_L(db_updater, db, "Writes to", "sync, PSQL")
Rel(db_updater, static_, "Get stops, lines andshapes")
@enduml
```
## Back-end
Conventional but efficient:
- [FastAPI](https://fastapi.tiangolo.com/): _FastAPI is a modern, fast (high-performance), web framework for building
APIs with Python 3.8+ based on standard Python type hints._
- [Pydantic 2](https://docs.pydantic.dev/latest/): _Pydantic is the most widely used data validation library for
Python._
- [Sqlalchemy 2](https://www.sqlalchemy.org/): _SQLAlchemy is the Python SQL toolkit and Object Relational Mapper that
gives application developers the full power and flexibility of SQL._
The [Msgspec](https://github.com/jcrist/msgspec) library is used to serialize JSON objects collected from IDFM API
endpoints, this library being faster than Pydantic.
## Front-end
The component is developed using the [SolidJS](https://www.solidjs.com/) library. It brings the following interesting specificities:
- Share with React the same programmatic structures and support for component.
- Fine-grained reactivity architecture: no virtual DOM used to update the components, the browser DOM is directly
updated by SolidJS.
The front-end tries to be as close as possible to the design defined by the IDFM for the displays deployed by the
transport operators in Ile-de-France. These specifications are public and available here:
[PRIM-IDFM](https://prim.iledefrance-mobilites.fr/en/chartes-et-prescriptions).
# TODO
## Features
- [ ] Integration with [Matrix.org](https://matrix.org/ecosystem/integrations/) ecosystem: make the app able to send message to a room when a bus/train/metro will
arriving in X minutes.
- [ ] Add the capability for the users to pin his/her favorite stops.
- [ ] Add the address to the stop location.
## Docs
- [ ] Describe how to build the front-end and back-end docker images.
- [ ] Describe how to deploy them using `docker compose`.
- [ ] Add back-end API description ([openAPI](https://www.openapis.org/)) + generate documentation
([mkdoc](https://www.mkdocs.org/), [Redoc](https://github.com/Redocly/redoc)). The best would be to build and
deploy it (as docker images) using CI/CD... need to test [Agola](https://agola.io/), [Jaypore
CI](https://www.jayporeci.in/) (pipelines configured in Python),
[Woodpecker](https://woodpecker-ci.org/docs/intro) or [Drone](https://www.drone.io/) before.
## Front-end
- [ ] Add unit tests.
- [ ] Make the StopNameInput component liquid.
- [ ] Liquid to responsive Design.
## Back-end
- [ ] Add unit tests.
- [ ] Use [alembic](https://alembic.sqlalchemy.org/en/latest/) to manage the future updates of the database schemas.
- [ ] Add the capability to reload the application configuration on configuration file update (e.g.: credential update by the vault).
- [ ] Rework how the database is updated with the IDFM data: For now the database is cleaned before refilling. It could
be useful to avoid to empty the database and only apply deltas (update/add/remove rows).
- [ ] Could be nice to compare FastAPI with [Litestar](https://litestar.dev/).
<!-- LocalWords: specificities
-->

View File

@@ -0,0 +1,33 @@
when:
- event: pull_request
branch:
exclude: [master, release/*]
steps:
- name: prepare
image: python:3.12-alpine
commands: |
cd ./backend
python3 -m venv local
wget -O - -q https://raw.githubusercontent.com/reviewdog/reviewdog/master/install.sh | sh -s latest -b ./local/bin/
source ./local/bin/activate
pip install poetry
poetry install --only=linters --no-root
- name: ruff
image: python:3.12-alpine
failure: ignore
secrets: ["reviewdog_gitea_api_token", "gitea_address"]
commands: |
cd ./backend
source ./local/bin/activate
poetry run ruff --output-format sarif . | ./local/bin/reviewdog -f sarif -reporter gitea-pr-review -filter-mode nofilter
- name: mypy
image: python:3.12-alpine
failure: ignore
secrets: ["reviewdog_gitea_api_token", "gitea_address"]
commands: |
cd ./backend
source ./local/bin/activate
poetry run mypy --no-incremental . | ./local/bin/reviewdog -f mypy -reporter gitea-pr-review -filter-mode nofilter

View File

@@ -9,7 +9,7 @@ ENV POETRY_NO_INTERACTION=1 \
WORKDIR /app WORKDIR /app
COPY ./pyproject.toml /app COPY pyproject.toml /app
RUN poetry install --only=main --no-root && \ RUN poetry install --only=main --no-root && \
rm -rf ${POETRY_CACHE_DIR} rm -rf ${POETRY_CACHE_DIR}
@@ -29,10 +29,8 @@ env VIRTUAL_ENV=/app/.venv \
COPY --from=builder ${VIRTUAL_ENV} ${VIRTUAL_ENV} COPY --from=builder ${VIRTUAL_ENV} ${VIRTUAL_ENV}
COPY backend /app/backend COPY api /app/api
COPY dependencies.py /app COPY config.sample.yaml .
COPY config.sample.yaml /app COPY api_server.py .
COPY routers/ /app/routers
COPY main.py /app
CMD ["python", "./main.py"] CMD ["./api_server.py"]

View File

@@ -34,10 +34,8 @@ env VIRTUAL_ENV=/app/.venv \
COPY --from=builder ${VIRTUAL_ENV} ${VIRTUAL_ENV} COPY --from=builder ${VIRTUAL_ENV} ${VIRTUAL_ENV}
COPY backend /app/backend COPY api /app/api
COPY dependencies.py /app COPY config.sample.yaml .
COPY config.sample.yaml /app COPY db_updater.py .
COPY config.local.yaml /app
COPY db_updater /app/db_updater
CMD ["python", "-m", "db_updater.fill_db"] CMD ["./db_updater.py"]

View File

@@ -0,0 +1,9 @@
# Architecture
The following schema shows the components used to satisfy the `api` service:
![API](./docs/medias/hubble-ui_api.png)

View File

@@ -61,9 +61,6 @@ class Database:
while not ret: while not ret:
try: try:
async with self._async_engine.begin() as session: async with self._async_engine.begin() as session:
await session.execute(
text("CREATE EXTENSION IF NOT EXISTS pg_trgm;")
)
if clear_static_data: if clear_static_data:
await session.run_sync(Base.metadata.drop_all) await session.run_sync(Base.metadata.drop_all)
await session.run_sync(Base.metadata.create_all) await session.run_sync(Base.metadata.create_all)

View File

@@ -4,9 +4,9 @@ from fastapi_cache.backends.redis import RedisBackend
from redis import asyncio as aioredis from redis import asyncio as aioredis
from yaml import safe_load from yaml import safe_load
from backend.db import db from .db import db
from backend.idfm_interface.idfm_interface import IdfmInterface from .idfm_interface.idfm_interface import IdfmInterface
from backend.settings import CacheSettings, Settings from .settings import CacheSettings, Settings
CONFIG_PATH = environ.get("CONFIG_PATH", "./config.sample.yaml") CONFIG_PATH = environ.get("CONFIG_PATH", "./config.sample.yaml")

View File

@@ -7,9 +7,9 @@ from aiohttp import ClientSession
from msgspec import ValidationError from msgspec import ValidationError
from msgspec.json import Decoder from msgspec.json import Decoder
from .idfm_types import Destinations as IdfmDestinations, IdfmResponse, IdfmState
from ..db import Database from ..db import Database
from ..models import Line, Stop, StopArea from ..models import Line, Stop, StopArea
from .idfm_types import Destinations as IdfmDestinations, IdfmResponse, IdfmState
class IdfmInterface: class IdfmInterface:
@@ -31,6 +31,10 @@ class IdfmInterface:
async def startup(self) -> None: async def startup(self) -> None:
... ...
@staticmethod
def _format_line_id(line_id: str) -> int:
return int(line_id[1:] if line_id[0] == "C" else line_id)
async def render_line_picto(self, line: Line) -> tuple[None | str, None | str]: async def render_line_picto(self, line: Line) -> tuple[None | str, None | str]:
line_picto_path = line_picto_format = None line_picto_path = line_picto_format = None
target = f"/tmp/{line.id}_repr" target = f"/tmp/{line.id}_repr"
@@ -81,7 +85,6 @@ class IdfmInterface:
if (stop := await Stop.get_by_id(stop_id)) is not None: if (stop := await Stop.get_by_id(stop_id)) is not None:
expected_stop_ids = {stop.id} expected_stop_ids = {stop.id}
elif (stop_area := await StopArea.get_by_id(stop_id)) is not None: elif (stop_area := await StopArea.get_by_id(stop_id)) is not None:
expected_stop_ids = {stop.id for stop in stop_area.stops} expected_stop_ids = {stop.id for stop in stop_area.stops}
else: else:
@@ -105,7 +108,8 @@ class IdfmInterface:
if ( if (
dst_names := journey.DestinationName dst_names := journey.DestinationName
) and monitored_stop_id in expected_stop_ids: ) and monitored_stop_id in expected_stop_ids:
line_id = journey.LineRef.value.split(":")[-2] raw_line_id = journey.LineRef.value.split(":")[-2]
line_id = IdfmInterface._format_line_id(raw_line_id)
destinations[line_id].add(dst_names[0].value) destinations[line_id].add(dst_names[0].value)
return destinations return destinations

View File

@@ -141,6 +141,8 @@ class StopAreaStopAssociationFields(Struct, kw_only=True):
artid: str | None = None artid: str | None = None
arrversion: str arrversion: str
zdcid: str zdcid: str
pdeid: str | None = None
pdeversion: int | None = None
version: int version: int
zdaid: str zdaid: str
zdaversion: str zdaversion: str
@@ -205,7 +207,7 @@ class Line(Struct):
Lines = dict[str, Line] Lines = dict[str, Line]
Destinations = dict[str, set[str]] Destinations = dict[int, set[str]]
# TODO: Set structs frozen # TODO: Set structs frozen

View File

@@ -35,6 +35,26 @@ if TYPE_CHECKING:
logger = getLogger(__name__) logger = getLogger(__name__)
# import cProfile
# import io
# import pstats
# import contextlib
# @contextlib.contextmanager
# def profiled():
# pr = cProfile.Profile()
# pr.enable()
# yield
# pr.disable()
# s = io.StringIO()
# ps = pstats.Stats(pr, stream=s).sort_stats("cumulative")
# ps.print_stats()
# # uncomment this to see who's calling what
# # ps.print_callers()
# print(s.getvalue())
class StopAreaStopAssociations(Base): class StopAreaStopAssociations(Base):
id = mapped_column(BigInteger, primary_key=True) id = mapped_column(BigInteger, primary_key=True)

View File

@@ -1,8 +1,8 @@
from fastapi import APIRouter, HTTPException from fastapi import APIRouter, HTTPException
from fastapi_cache.decorator import cache from fastapi_cache.decorator import cache
from backend.models import Line from ..models import Line
from backend.schemas import Line as LineSchema, TransportMode from ..schemas import Line as LineSchema, TransportMode
router = APIRouter(prefix="/line", tags=["line"]) router = APIRouter(prefix="/line", tags=["line"])

View File

@@ -5,16 +5,16 @@ from typing import Sequence
from fastapi import APIRouter, HTTPException from fastapi import APIRouter, HTTPException
from fastapi_cache.decorator import cache from fastapi_cache.decorator import cache
from backend.idfm_interface import Destinations as IdfmDestinations, TrainStatus from ..idfm_interface import Destinations as IdfmDestinations, TrainStatus
from backend.models import Stop, StopArea, StopShape from ..models import Stop, StopArea, StopShape
from backend.schemas import ( from ..schemas import (
NextPassage as NextPassageSchema, NextPassage as NextPassageSchema,
NextPassages as NextPassagesSchema, NextPassages as NextPassagesSchema,
Stop as StopSchema, Stop as StopSchema,
StopArea as StopAreaSchema, StopArea as StopAreaSchema,
StopShape as StopShapeSchema, StopShape as StopShapeSchema,
) )
from dependencies import idfm_interface from ..dependencies import idfm_interface
router = APIRouter(prefix="/stop", tags=["stop"]) router = APIRouter(prefix="/stop", tags=["stop"])

View File

@@ -53,8 +53,8 @@ class Line(BaseModel):
transportMode: TransportMode transportMode: TransportMode
backColorHexa: str backColorHexa: str
foreColorHexa: str foreColorHexa: str
operatorId: str operatorId: int
accessibility: IdfmState accessibility: IdfmState
visualSignsAvailable: IdfmState visualSignsAvailable: IdfmState
audibleSignsAvailable: IdfmState audibleSignsAvailable: IdfmState
stopIds: list[str] stopIds: list[int]

View File

@@ -9,7 +9,7 @@ class Stop(BaseModel):
town: str town: str
epsg3857_x: float epsg3857_x: float
epsg3857_y: float epsg3857_y: float
lines: list[str] lines: list[int]
class StopArea(BaseModel): class StopArea(BaseModel):
@@ -17,7 +17,7 @@ class StopArea(BaseModel):
name: str name: str
town: str town: str
type: StopAreaType type: StopAreaType
lines: list[str] # SNCF lines are linked to stop areas and not stops. lines: list[int] # SNCF lines are linked to stop areas and not stops.
stops: list[Stop] stops: list[Stop]

74
backend/api/settings.py Normal file
View File

@@ -0,0 +1,74 @@
from __future__ import annotations
from typing import Annotated
from pydantic import BaseModel, SecretStr
from pydantic.functional_validators import model_validator
from pydantic_settings import (
BaseSettings,
PydanticBaseSettingsSource,
SettingsConfigDict,
)
class HttpSettings(BaseModel):
host: str = "127.0.0.1"
port: int = 8080
cert: str | None = None
class DatabaseSettings(BaseModel):
name: str
host: str
port: int
driver: str = "postgresql+psycopg"
user: str
password: Annotated[SecretStr, check_user_password]
class CacheSettings(BaseModel):
enable: bool = False
host: str = "127.0.0.1"
port: int = 6379
user: str | None = None
password: Annotated[SecretStr | None, check_user_password] = None
@model_validator(mode="after")
def check_user_password(self) -> DatabaseSettings | CacheSettings:
if self.user is not None and self.password is None:
raise ValueError("user is set, password shall be set too.")
if self.password is not None and self.user is None:
raise ValueError("password is set, user shall be set too.")
return self
class TracingSettings(BaseModel):
enable: bool = False
class Settings(BaseSettings):
app_name: str
idfm_api_key: SecretStr
clear_static_data: bool
http: HttpSettings
db: DatabaseSettings
cache: CacheSettings
tracing: TracingSettings
model_config = SettingsConfigDict(env_prefix="CER__", env_nested_delimiter="__")
@classmethod
def settings_customise_sources(
cls,
settings_cls: type[BaseSettings],
init_settings: PydanticBaseSettingsSource,
env_settings: PydanticBaseSettingsSource,
dotenv_settings: PydanticBaseSettingsSource,
file_secret_settings: PydanticBaseSettingsSource,
) -> tuple[PydanticBaseSettingsSource, ...]:
return env_settings, init_settings, file_secret_settings

View File

@@ -4,7 +4,6 @@ import uvicorn
from contextlib import asynccontextmanager from contextlib import asynccontextmanager
from fastapi import FastAPI, Request from fastapi import FastAPI, Request
from fastapi.middleware.cors import CORSMiddleware from fastapi.middleware.cors import CORSMiddleware
from fastapi.staticfiles import StaticFiles
from fastapi_cache import FastAPICache from fastapi_cache import FastAPICache
from opentelemetry import trace from opentelemetry import trace
from opentelemetry.exporter.otlp.proto.http.trace_exporter import OTLPSpanExporter from opentelemetry.exporter.otlp.proto.http.trace_exporter import OTLPSpanExporter
@@ -13,19 +12,20 @@ from opentelemetry.sdk.resources import Resource, SERVICE_NAME
from opentelemetry.sdk.trace import TracerProvider from opentelemetry.sdk.trace import TracerProvider
from opentelemetry.sdk.trace.export import BatchSpanProcessor from opentelemetry.sdk.trace.export import BatchSpanProcessor
from backend.db import db from api.db import db
from dependencies import idfm_interface, redis_backend, settings from api.dependencies import idfm_interface, redis_backend, settings
from routers import line, stop from api.routers import line, stop
@asynccontextmanager @asynccontextmanager
async def lifespan(app: FastAPI): async def lifespan(app: FastAPI) -> None:
FastAPICache.init(redis_backend, prefix="api", enable=settings.cache.enable) FastAPICache.init(redis_backend, prefix="api", enable=settings.cache.enable)
await db.connect(settings.db, settings.clear_static_data) await db.connect(settings.db, settings.clear_static_data)
if settings.clear_static_data: if settings.clear_static_data:
await idfm_interface.startup() await idfm_interface.startup()
print("OK")
yield yield
await db.disconnect() await db.disconnect()
@@ -35,14 +35,12 @@ app = FastAPI(lifespan=lifespan)
app.add_middleware( app.add_middleware(
CORSMiddleware, CORSMiddleware,
allow_origins=["https://localhost:4443", "https://localhost:3000"], allow_origins=["http://carrramba.adrien.run", "https://carrramba.adrien.run"],
allow_credentials=True, allow_credentials=True,
allow_methods=["*"], allow_methods=["OPTIONS", "GET"],
allow_headers=["*"], allow_headers=["*"],
) )
app.mount("/widget", StaticFiles(directory="../frontend/", html=True), name="widget")
# The cache-control header entry is not managed properly by fastapi-cache: # The cache-control header entry is not managed properly by fastapi-cache:
# For now, a request with a cache-control set to no-cache # For now, a request with a cache-control set to no-cache
# is interpreted as disabling the use of the server cache. # is interpreted as disabling the use of the server cache.
@@ -60,7 +58,6 @@ async def fastapi_cache_issue_144_workaround(request: Request, call_next):
return await call_next(request) return await call_next(request)
app.include_router(line.router) app.include_router(line.router)
app.include_router(stop.router) app.include_router(stop.router)

View File

@@ -1,59 +0,0 @@
from typing import Any
from pydantic import BaseModel, BaseSettings, Field, root_validator, SecretStr
class HttpSettings(BaseModel):
host: str = "127.0.0.1"
port: int = 8080
cert: str | None = None
def check_user_password(cls, values: dict[str, Any]) -> dict[str, Any]:
user = values.get("user")
password = values.get("password")
if user is not None and password is None:
raise ValueError("user is set, password shall be set too.")
if password is not None and user is None:
raise ValueError("password is set, user shall be set too.")
return values
class DatabaseSettings(BaseModel):
name: str = "carrramba-encore-rate"
host: str = "127.0.0.1"
port: int = 5432
driver: str = "postgresql+psycopg"
user: str | None = None
password: SecretStr | None = None
_user_password_validation = root_validator(allow_reuse=True)(check_user_password)
class CacheSettings(BaseModel):
enable: bool = False
host: str = "127.0.0.1"
port: int = 6379
user: str | None = None
password: SecretStr | None = None
_user_password_validation = root_validator(allow_reuse=True)(check_user_password)
class TracingSettings(BaseModel):
enable: bool = False
class Settings(BaseSettings):
app_name: str
idfm_api_key: SecretStr = Field(..., env="IDFM_API_KEY")
clear_static_data: bool = Field(False, env="CLEAR_STATIC_DATA")
http: HttpSettings = HttpSettings()
db: DatabaseSettings = DatabaseSettings()
cache: CacheSettings = CacheSettings()
tracing: TracingSettings = TracingSettings()

View File

@@ -2,7 +2,7 @@ app_name: carrramba-encore-rate
clear_static_data: false clear_static_data: false
http: http:
host: 0.0.0.0 host: 127.0.0.1
port: 8080 port: 8080
cert: ./config/cert.pem cert: ./config/cert.pem

View File

@@ -10,9 +10,7 @@ db:
name: carrramba-encore-rate name: carrramba-encore-rate
host: postgres host: postgres
port: 5432 port: 5432
driver: postgresql+psycopg
user: cer user: cer
password: cer_password
cache: cache:
enable: true enable: true

View File

@@ -16,9 +16,9 @@ from shapefile import Reader as ShapeFileReader, ShapeRecord # type: ignore
from tqdm import tqdm from tqdm import tqdm
from yaml import safe_load from yaml import safe_load
from backend.db import Base, db, Database from api.db import Base, db, Database
from backend.models import ConnectionArea, Line, LinePicto, Stop, StopArea, StopShape from api.models import ConnectionArea, Line, LinePicto, Stop, StopArea, StopShape
from backend.idfm_interface.idfm_types import ( from api.idfm_interface.idfm_types import (
ConnectionArea as IdfmConnectionArea, ConnectionArea as IdfmConnectionArea,
IdfmLineState, IdfmLineState,
Line as IdfmLine, Line as IdfmLine,
@@ -31,8 +31,8 @@ from backend.idfm_interface.idfm_types import (
StopLineAsso as IdfmStopLineAsso, StopLineAsso as IdfmStopLineAsso,
TransportMode, TransportMode,
) )
from backend.idfm_interface.ratp_types import Picto as RatpPicto from api.idfm_interface.ratp_types import Picto as RatpPicto
from backend.settings import Settings from api.settings import Settings
CONFIG_PATH = environ.get("CONFIG_PATH", "./config.sample.yaml") CONFIG_PATH = environ.get("CONFIG_PATH", "./config.sample.yaml")

View File

@@ -9,7 +9,7 @@ readme = "README.md"
python = "^3.11" python = "^3.11"
aiohttp = "^3.8.3" aiohttp = "^3.8.3"
aiofiles = "^22.1.0" aiofiles = "^22.1.0"
fastapi = "^0.95.0" fastapi = "^0.103.0"
uvicorn = "^0.20.0" uvicorn = "^0.20.0"
msgspec = "^0.12.0" msgspec = "^0.12.0"
opentelemetry-instrumentation-fastapi = "^0.38b0" opentelemetry-instrumentation-fastapi = "^0.38b0"
@@ -23,11 +23,13 @@ sqlalchemy = "^2.0.12"
psycopg = "^3.1.9" psycopg = "^3.1.9"
pyyaml = "^6.0" pyyaml = "^6.0"
fastapi-cache2 = {extras = ["redis"], version = "^0.2.1"} fastapi-cache2 = {extras = ["redis"], version = "^0.2.1"}
pydantic-settings = "^2.0.3"
ruff = "^0.2.1"
[tool.poetry.group.db_updater.dependencies] [tool.poetry.group.db_updater.dependencies]
aiofiles = "^22.1.0" aiofiles = "^22.1.0"
aiohttp = "^3.8.3" aiohttp = "^3.8.3"
fastapi = "^0.95.0" fastapi = "^0.103.0"
msgspec = "^0.12.0" msgspec = "^0.12.0"
opentelemetry-instrumentation-fastapi = "^0.38b0" opentelemetry-instrumentation-fastapi = "^0.38b0"
opentelemetry-instrumentation-sqlalchemy = "^0.38b0" opentelemetry-instrumentation-sqlalchemy = "^0.38b0"
@@ -41,35 +43,46 @@ pyyaml = "^6.0"
sqlalchemy = "^2.0.12" sqlalchemy = "^2.0.12"
sqlalchemy-utils = "^0.41.1" sqlalchemy-utils = "^0.41.1"
tqdm = "^4.65.0" tqdm = "^4.65.0"
pydantic-settings = "^2.0.3"
[build-system] [build-system]
requires = ["poetry-core"] requires = ["poetry-core"]
build-backend = "poetry.core.masonry.api" build-backend = "poetry.core.masonry.api"
[tool.poetry.group.dev.dependencies] [tool.poetry.group.dev.dependencies]
pylsp-mypy = "^0.6.2" pylsp-mypy = "^0.6.2"
mccabe = "^0.7.0"
rope = "^1.3.0"
python-lsp-black = "^1.2.1" python-lsp-black = "^1.2.1"
black = "^22.10.0"
types-aiofiles = "^22.1.0.2"
wrapt = "^1.14.1" wrapt = "^1.14.1"
pydocstyle = "^6.2.2"
dill = "^0.3.6" dill = "^0.3.6"
python-lsp-ruff = "^1.0.5" python-lsp-ruff = "^1.0.5"
python-lsp-server = "^1.7.1" python-lsp-server = "^1.7.1"
autopep8 = "^2.0.1"
pyflakes = "^3.0.1"
yapf = "^0.32.0"
whatthepatch = "^1.0.4"
mypy = "^1.0.0"
icecream = "^2.1.3" icecream = "^2.1.3"
[tool.poetry.group.linters.dependencies]
autopep8 = "^2.0.1"
black = "^22.10.0"
mccabe = "^0.7.0"
mypy = "^1.0.0"
pydocstyle = "^6.2.2"
pyflakes = "^3.0.1"
rope = "^1.3.0"
ruff = "^0.2.1"
types-aiofiles = "^22.1.0.2"
types-sqlalchemy-utils = "^1.0.1" types-sqlalchemy-utils = "^1.0.1"
types-pyyaml = "^6.0.12.9" types-pyyaml = "^6.0.12.9"
types-tqdm = "^4.65.0.1" types-tqdm = "^4.65.0.1"
whatthepatch = "^1.0.4"
yapf = "^0.32.0"
sqlalchemy = "^2.0.26"
[tool.mypy] [tool.mypy]
plugins = "sqlalchemy.ext.mypy.plugin" plugins = "sqlalchemy.ext.mypy.plugin"
exclude = ['docker', 'docs']
strict = true
[tool.black] [tool.black]
target-version = ['py311'] target-version = ['py311']

View File

@@ -32,6 +32,7 @@
"matrix-widget-api": "^1.1.1", "matrix-widget-api": "^1.1.1",
"ol": "^7.3.0", "ol": "^7.3.0",
"solid-js": "^1.6.6", "solid-js": "^1.6.6",
"solid-transition-group": "^0.0.10" "solid-transition-group": "^0.0.10",
"solidjs-lazily": "^0.1.2"
} }
} }

View File

@@ -1,15 +1,12 @@
import { Component, createSignal } from 'solid-js'; import { Component, createSignal, onCleanup, onMount } from 'solid-js';
import { IVisibilityActionRequest, MatrixCapabilities, WidgetApi, WidgetApiToWidgetAction } from 'matrix-widget-api'; // import { IVisibilityActionRequest, MatrixCapabilities, WidgetApi, WidgetApiToWidgetAction } from 'matrix-widget-api';
import { HopeProvider } from "@hope-ui/solid";
import { BusinessDataProvider } from './businessData'; import { BusinessDataProvider } from './businessData';
import { AppContextProvider } from './appContext'; import { AppContextProvider } from './appContext';
import { PassagesDisplay } from './passagesDisplay'; import { PassagesDisplay } from './passagesDisplay';
import { StopsSearchMenu } from './stopsSearchMenu/stopsSearchMenu'; import { StopsSearchMenu } from './stopsSearchMenu/stopsSearchMenu';
import "./App.scss"; import "./App.scss";
import { onCleanup, onMount } from 'solid-js';
function parseFragment() { function parseFragment() {
@@ -28,21 +25,21 @@ const App: Component = () => {
console.log("App: widgetId:" + widgetId); console.log("App: widgetId:" + widgetId);
console.log("App: userId:" + userId); console.log("App: userId:" + userId);
const api = new WidgetApi(widgetId != null ? widgetId : undefined); // const api = new WidgetApi(widgetId != null ? widgetId : undefined);
api.requestCapability(MatrixCapabilities.AlwaysOnScreen); // api.requestCapability(MatrixCapabilities.AlwaysOnScreen);
api.start(); // api.start();
api.on("ready", function() { // api.on("ready", function() {
console.log("App: widget API is READY !!!!"); // console.log("App: widget API is READY !!!!");
}); // });
// Seems to don´t be used... // Seems to don´t be used...
api.on(`action:${WidgetApiToWidgetAction.UpdateVisibility}`, (ev: CustomEvent<IVisibilityActionRequest>) => { // api.on(`action:${WidgetApiToWidgetAction.UpdateVisibility}`, (ev: CustomEvent<IVisibilityActionRequest>) => {
console.log("App: Visibility change"); // console.log("App: Visibility change");
ev.preventDefault(); // we're handling it, so stop the widget API from doing something. // ev.preventDefault(); // we're handling it, so stop the widget API from doing something.
console.log("App: ", ev.detail); // custom handling here // console.log("App: ", ev.detail); // custom handling here
/* api.transport.reply(ev.detail, <IWidgetApiRequestEmptyData>{}); */ // /* api.transport.reply(ev.detail, <IWidgetApiRequestEmptyData>{}); */
api.transport.reply(ev.detail, {}); // api.transport.reply(ev.detail, {});
}); // });
createSignal({ createSignal({
height: window.innerHeight, height: window.innerHeight,
@@ -71,10 +68,8 @@ const App: Component = () => {
window.removeEventListener('resize', onResize); window.removeEventListener('resize', onResize);
}) })
return ( return <BusinessDataProvider>
<BusinessDataProvider>
<AppContextProvider> <AppContextProvider>
<HopeProvider>
<div class="App"> <div class="App">
<div class="panel"> <div class="panel">
<StopsSearchMenu /> <StopsSearchMenu />
@@ -83,10 +78,8 @@ const App: Component = () => {
<PassagesDisplay /> <PassagesDisplay />
</div> </div>
</div> </div>
</HopeProvider>
</AppContextProvider> </AppContextProvider>
</BusinessDataProvider> </BusinessDataProvider>;
);
}; };
export default App; export default App;

View File

@@ -6,7 +6,7 @@ import { Stop } from './types';
export interface AppContextStore { export interface AppContextStore {
getDisplayedStops: () => Stop[]; getDisplayedStops: () => Stop[];
setDisplayedStops: (stops: Stop[]) => void; setDisplayedStops: (stops: Stop[]) => void;
}; }
export const AppContextContext = createContext<AppContextStore>(); export const AppContextContext = createContext<AppContextStore>();
@@ -26,10 +26,7 @@ export function AppContextProvider(props: { children: JSX.Element }) {
const setDisplayedStops = (stops: Stop[]): void => { const setDisplayedStops = (stops: Stop[]): void => {
console.log("setDisplayedStops=", stops); console.log("setDisplayedStops=", stops);
// setStore((s: Store) => {
setStore('displayedStops', stops); setStore('displayedStops', stops);
// return s;
// });
} }
return ( return (
@@ -39,5 +36,4 @@ export function AppContextProvider(props: { children: JSX.Element }) {
{props.children} {props.children}
</AppContextContext.Provider> </AppContextContext.Provider>
); );
}
};

View File

@@ -6,6 +6,7 @@ import { Line, Lines, Passage, Passages, Stop, StopShape, StopShapes, Stops } fr
export type StopDestinations = Record<string, string[]>; export type StopDestinations = Record<string, string[]>;
export interface BusinessDataStore { export interface BusinessDataStore {
getLine: (lineId: string) => Promise<Line>; getLine: (lineId: string) => Promise<Line>;
getLinePassages: (lineId: string) => Record<string, Passage[]>; getLinePassages: (lineId: string) => Record<string, Passage[]>;
@@ -23,13 +24,12 @@ export interface BusinessDataStore {
getStopDestinations: (stopId: number) => Promise<StopDestinations | undefined>; getStopDestinations: (stopId: number) => Promise<StopDestinations | undefined>;
getStopShape: (stopId: number) => Promise<StopShape | undefined>; getStopShape: (stopId: number) => Promise<StopShape | undefined>;
}; }
export const BusinessDataContext = createContext<BusinessDataStore>(); export const BusinessDataContext = createContext<BusinessDataStore>();
export function BusinessDataProvider(props: { children: JSX.Element }) { export function BusinessDataProvider(props: { children: JSX.Element }) {
const [serverUrl] = createSignal<string>("https://carrramba.adrien.run/api");
const [serverUrl] = createSignal<string>("https://localhost:4443");
type Store = { type Store = {
lines: Lines; lines: Lines;
@@ -121,7 +121,7 @@ export function BusinessDataProvider(props: { children: JSX.Element }) {
for (const lineId of Object.keys(passages)) { for (const lineId of Object.keys(passages)) {
const newLinePassages = passages[lineId]; const newLinePassages = passages[lineId];
const linePassages = storePassages[lineId]; const linePassages = storePassages[lineId];
if (linePassages === undefined) { if (linePassages === undefined || Object.keys(linePassages).length == 0) {
setStore('passages', lineId, newLinePassages); setStore('passages', lineId, newLinePassages);
} }
else { else {
@@ -160,7 +160,7 @@ ${linePassagesDestination.length} here... refresh all them.`);
const clearPassages = (): void => { const clearPassages = (): void => {
setStore((s: Store): Store => { setStore((s: Store): Store => {
for (const lineId of Object.keys(s.passages)) { for (const lineId of Object.keys(s.passages)) {
setStore('passages', lineId, undefined); setStore('passages', lineId, {});
} }
return s; return s;
}); });
@@ -232,21 +232,9 @@ ${linePassagesDestination.length} here... refresh all them.`);
getLine, getLinePassages, getLineDestinations, getDestinationPassages, passages, getPassagesLineIds, getLine, getLinePassages, getLineDestinations, getDestinationPassages, passages, getPassagesLineIds,
refreshPassages, addPassages, clearPassages, refreshPassages, addPassages, clearPassages,
getStop, getStopDestinations, getStopShape, searchStopByName getStop, getStopDestinations, getStopShape, searchStopByName
}}> }}>
{props.children} {props.children}
</BusinessDataContext.Provider> </BusinessDataContext.Provider>
); );
} }
export interface BusinessDataStore {
getLine: (lineId: string) => Promise<Line>;
getLinePassages: (lineId: string) => Record<string, Passage[]>;
passages: () => Passages;
refreshPassages: (stopId: number) => Promise<void>;
addPassages: (passages: Passages) => void;
clearPassages: () => void;
getStop: (stopId: number) => Stop | undefined;
searchStopByName: (name: string) => Promise<Stops>;
};

View File

@@ -4,7 +4,8 @@ import { VoidComponent } from "solid-js";
export const IconHamburgerMenu: VoidComponent<{}> = () => { export const IconHamburgerMenu: VoidComponent<{}> = () => {
return ( return (
<svg class="iconHamburgerMenu" viewBox="0 0 15 15"> <svg class="iconHamburgerMenu" viewBox="0 0 15 15">
<path d="M1.5 3C1.22386 3 1 3.22386 1 3.5C1 3.77614 1.22386 4 1.5 4H13.5C13.7761 4 14 3.77614 14 3.5C14 3.22386 <path
d="M1.5 3C1.22386 3 1 3.22386 1 3.5C1 3.77614 1.22386 4 1.5 4H13.5C13.7761 4 14 3.77614 14 3.5C14 3.22386
13.7761 3 13.5 3H1.5ZM1 7.5C1 7.22386 1.22386 7 1.5 7H13.5C13.7761 7 14 7.22386 14 7.5C14 7.77614 13.7761 8 13.5 13.7761 3 13.5 3H1.5ZM1 7.5C1 7.22386 1.22386 7 1.5 7H13.5C13.7761 7 14 7.22386 14 7.5C14 7.77614 13.7761 8 13.5
8H1.5C1.22386 8 1 7.77614 1 7.5ZM1 11.5C1 11.2239 1.22386 11 1.5 11H13.5C13.7761 11 14 11.2239 14 11.5C14 11.7761 8H1.5C1.22386 8 1 7.77614 1 7.5ZM1 11.5C1 11.2239 1.22386 11 1.5 11H13.5C13.7761 11 14 11.2239 14 11.5C14 11.7761
13.7761 12 13.5 12H1.5C1.22386 12 1 11.7761 1 11.5Z" 13.7761 12 13.5 12H1.5C1.22386 12 1 11.7761 1 11.5Z"

View File

@@ -14,6 +14,7 @@
} }
html, body { html, body {
height: 100vh;
aspect-ratio: 16/9; aspect-ratio: 16/9;
margin: 0; margin: 0;

View File

@@ -1,7 +1,6 @@
import { createContext, createEffect, createResource, createSignal, For, JSX, ParentComponent, Show, useContext, VoidComponent } from "solid-js"; import { createContext, createEffect, createResource, createSignal, For, JSX, ParentComponent, Show, useContext, VoidComponent } from "solid-js";
import { createStore } from "solid-js/store"; import { createStore } from "solid-js/store";
import { createDateNow } from "@solid-primitives/date"; import { createDateNow } from "@solid-primitives/date";
import { IconButton, Menu, MenuTrigger, MenuContent, MenuItem } from "@hope-ui/solid";
import { format } from "date-fns"; import { format } from "date-fns";
import { BusinessDataContext, BusinessDataStore } from "./businessData"; import { BusinessDataContext, BusinessDataStore } from "./businessData";
@@ -9,7 +8,6 @@ import { AppContextContext, AppContextStore } from "./appContext";
import { getTransportModeSrc, PositionedPanel } from "./utils"; import { getTransportModeSrc, PositionedPanel } from "./utils";
import { PassagesPanel } from "./passagesPanel"; import { PassagesPanel } from "./passagesPanel";
import { IconHamburgerMenu } from './extra/iconHamburgerMenu';
import "./passagesDisplay.scss"; import "./passagesDisplay.scss";
@@ -25,7 +23,7 @@ interface PassagesDisplayStore {
getDisplayedPanelId: () => number; getDisplayedPanelId: () => number;
setDisplayedPanelId: (panelId: number) => void; setDisplayedPanelId: (panelId: number) => void;
}; }
const PassagesDisplayContext = createContext<PassagesDisplayStore>(); const PassagesDisplayContext = createContext<PassagesDisplayStore>();
@@ -91,7 +89,6 @@ const Header: VoidComponent<{ title: string }> = (props) => {
return <div />; return <div />;
const { getLine, passages } = businessDataStore; const { getLine, passages } = businessDataStore;
const { isPassagesRefreshEnabled, togglePassagesRefresh } = passagesDisplayStore;
const [dateNow] = createDateNow(1000); const [dateNow] = createDateNow(1000);
@@ -113,16 +110,14 @@ const Header: VoidComponent<{ title: string }> = (props) => {
setLinesIds(Object.keys(passages())); setLinesIds(Object.keys(passages()));
}); });
return ( return <div class="header">
<div class="header">
<Show when={transportModeUrls() !== undefined} > <Show when={transportModeUrls() !== undefined} >
<For each={transportModeUrls()}> <For each={transportModeUrls()}>
{(url) => {(url) =>
<div class="transportMode"> <div class="transportMode">
<img src={url} /> <img src={url} />
</div> </div>
} }</For>
</For>
</Show> </Show>
<div class="title"> <div class="title">
<svg viewBox="0 0 1260 50"> <svg viewBox="0 0 1260 50">
@@ -132,15 +127,6 @@ const Header: VoidComponent<{ title: string }> = (props) => {
</svg> </svg>
</div> </div>
<div class="menu"> <div class="menu">
<Menu>
<MenuTrigger
as={IconButton}
icon=<IconHamburgerMenu />
/>
<MenuContent>
<MenuItem onSelect={() => togglePassagesRefresh()}>{isPassagesRefreshEnabled() ? "Disable" : "Enable"}</MenuItem>
</MenuContent>
</Menu>
</div> </div>
<div class="clock"> <div class="clock">
<svg viewBox="0 0 115 43"> <svg viewBox="0 0 115 43">
@@ -149,8 +135,7 @@ const Header: VoidComponent<{ title: string }> = (props) => {
</text> </text>
</svg> </svg>
</div> </div>
</div > </div>;
);
}; };
const Footer: VoidComponent<{}> = () => { const Footer: VoidComponent<{}> = () => {
@@ -176,8 +161,7 @@ const Footer: VoidComponent<{}> = () => {
</svg> </svg>
</div> </div>
); );
}} }}</For>
</For>
</div> </div>
); );
} }
@@ -286,13 +270,11 @@ export const PassagesDisplay: ParentComponent = () => {
const syncPeriodMsec = 20 * 1000; const syncPeriodMsec = 20 * 1000;
const panelSwitchPeriodMsec = 4 * 1000; const panelSwitchPeriodMsec = 4 * 1000;
return ( return <div class="passagesDisplay">
<div class="passagesDisplay">
<PassagesDisplayProvider> <PassagesDisplayProvider>
<Header title="Prochains passages" /> <Header title="Prochains passages" />
<Body maxPassagesPerPanel={MAX_PASSAGES_PER_PANEL} syncPeriodMsec={syncPeriodMsec} panelSwitchPeriodMsec={panelSwitchPeriodMsec} /> <Body maxPassagesPerPanel={MAX_PASSAGES_PER_PANEL} syncPeriodMsec={syncPeriodMsec} panelSwitchPeriodMsec={panelSwitchPeriodMsec} />
<Footer /> <Footer />
</PassagesDisplayProvider> </PassagesDisplayProvider>
</div> </div>;
);
}; };

View File

@@ -14,15 +14,13 @@ import "./passagesPanel.scss";
const UnavailablePassage: VoidComponent<{ style: string }> = (props) => { const UnavailablePassage: VoidComponent<{ style: string }> = (props) => {
const textStyle = { fill: "#000000" }; const textStyle = { fill: "#000000" };
return ( return <div class={props.style}>
<div class={props.style}>
<svg viewBox="0 0 230 110"> <svg viewBox="0 0 230 110">
<text x="100%" y="26" font-size="25" text-anchor="end" style={textStyle}>Information</text> <text x="100%" y="26" font-size="25" text-anchor="end" style={textStyle}>Information</text>
<text x="100%" y="63" font-size="25" text-anchor="end" style={textStyle}>non</text> <text x="100%" y="63" font-size="25" text-anchor="end" style={textStyle}>non</text>
<text x="100%" y="100" font-size="25" text-anchor="end" style={textStyle}>disponible</text> <text x="100%" y="100" font-size="25" text-anchor="end" style={textStyle}>disponible</text>
</svg> </svg>
</div> </div>;
);
} }
const Platform: VoidComponent<{ name: string }> = (props) => { const Platform: VoidComponent<{ name: string }> = (props) => {
@@ -43,14 +41,12 @@ const Platform: VoidComponent<{ name: string }> = (props) => {
} }
}); });
return ( return <svg class="platform" viewBox={`0 0 ${viewBoxWidthPx} 40`}>
<svg class="platform" viewBox={`0 0 ${viewBoxWidthPx} 40`}>
<rect ref={rectRef} x="0" y="0" height="100%" rx="9" ry="9" /> <rect ref={rectRef} x="0" y="0" height="100%" rx="9" ry="9" />
<text ref={textRef} x="100%" y="55%" dominant-baseline="middle" text-anchor="end" font-size="25" style={{ fill: "#ffffff" }}> <text ref={textRef} x="100%" y="55%" dominant-baseline="middle" text-anchor="end" font-size="25" style={{ fill: "#ffffff" }}>
QUAI {props.name} QUAI {props.name}
</text> </text>
</svg> </svg>;
);
} }
const TtwPassage: VoidComponent<{ const TtwPassage: VoidComponent<{
@@ -88,8 +84,7 @@ const TtwPassage: VoidComponent<{
</Motion.text> </Motion.text>
</svg>; </svg>;
return ( return <Show when={passage !== undefined} fallback={<UnavailablePassage style={props.fallbackStyle} />}>
<Show when={passage !== undefined} fallback=<UnavailablePassage style={props.fallbackStyle} />>
<Show <Show
when={passage.arrivalPlatformName !== null} when={passage.arrivalPlatformName !== null}
fallback={ fallback={
@@ -101,8 +96,7 @@ const TtwPassage: VoidComponent<{
<Platform name={passage.arrivalPlatformName} /> <Platform name={passage.arrivalPlatformName} />
</div> </div>
</Show> </Show>
</Show > </Show>;
);
}); });
} }
@@ -122,8 +116,7 @@ const DestinationPassages: VoidComponent<{ line: Line, destination: string }> =
// const trafficStatusStyle = { fill: trafficStatusColor.get(props.line.trafficStatus) }; // const trafficStatusStyle = { fill: trafficStatusColor.get(props.line.trafficStatus) };
const trafficStatusStyle = { fill: trafficStatusColor.get(TrafficStatus.UNKNOWN) }; const trafficStatusStyle = { fill: trafficStatusColor.get(TrafficStatus.UNKNOWN) };
return ( return <div class="line">
<div class="line">
<div class="transportMode"> <div class="transportMode">
{renderLineTransportMode(props.line)} {renderLineTransportMode(props.line)}
</div> </div>
@@ -142,8 +135,7 @@ const DestinationPassages: VoidComponent<{ line: Line, destination: string }> =
<TtwPassage line={props.line} destination={props.destination} index={1} <TtwPassage line={props.line} destination={props.destination} index={1}
style="secondPassage" withPlatformStyle="withPlatformSecondPassage" style="secondPassage" withPlatformStyle="withPlatformSecondPassage"
fontSize={45} fallbackStyle="unavailableSecondPassage" /> fontSize={45} fallbackStyle="unavailableSecondPassage" />
</div > </div>;
);
} }
export type PassagesPanelComponentProps = ParentProps & { stopId: number, lineIds: string[], show: boolean }; export type PassagesPanelComponentProps = ParentProps & { stopId: number, lineIds: string[], show: boolean };
@@ -162,19 +154,16 @@ export const PassagesPanel: PassagesPanelComponent = (props) => {
} }
const [lines] = createResource<Line[], string[]>(props.lineIds, getLines); const [lines] = createResource<Line[], string[]>(props.lineIds, getLines);
return ( return <div classList={{ "passagesContainer": true, "displayed": props.show }} >
<div classList={{ ["passagesContainer"]: true, ["displayed"]: props.show }} >
<Show when={lines() !== undefined} > <Show when={lines() !== undefined} >
<For each={lines()}> <For each={lines()}>
{(line) => {(line) =>
<Show when={getLineDestinations(line.id) !== undefined}> <Show when={getLineDestinations(line.id) !== undefined}>
<For each={getLineDestinations(line.id)}> <For each={getLineDestinations(line.id)}>{
{(destination) => <DestinationPassages line={line} destination={destination} />} (destination) => <DestinationPassages line={line} destination={destination} />
</For> }</For>
</Show> </Show>
} }</For>
</For>
</Show> </Show>
</div > </div>;
);
} }

View File

@@ -1,8 +1,8 @@
import { batch, createContext, JSX } from 'solid-js'; import { createContext, JSX } from 'solid-js';
import { createStore } from 'solid-js/store'; import { createStore } from 'solid-js/store';
import { Marker as LeafletMarker } from 'leaflet'; import { Marker as LeafletMarker } from 'leaflet';
import { Stop, Stops } from './types'; import { Stop } from './types';
export type ByStopIdMarkers = Record<number, LeafletMarker[] | undefined>; export type ByStopIdMarkers = Record<number, LeafletMarker[] | undefined>;
@@ -52,9 +52,7 @@ export function SearchProvider(props: { children: JSX.Element }) {
setStore('markers', stopId, markers); setStore('markers', stopId, markers);
} }
return ( return <SearchContext.Provider value={{ getFoundStops, setFoundStops, getDisplayedStops, setDisplayedStops, addMarkers }}>
<SearchContext.Provider value={{ getFoundStops, setFoundStops, getDisplayedStops, setDisplayedStops, addMarkers }}>
{props.children} {props.children}
</SearchContext.Provider> </SearchContext.Provider>;
);
} }

View File

@@ -41,7 +41,7 @@ export const Map: ParentComponent<{}> = () => {
// TODO: Set padding according to the marker design. // TODO: Set padding according to the marker design.
const fitPointsPadding = [50, 50, 50, 50]; const fitPointsPadding = [50, 50, 50, 50];
let mapDiv: HTMLDivElement | undefined = undefined; let mapDiv: HTMLDivElement | undefined;
let popup: StopPopup | undefined = undefined; let popup: StopPopup | undefined = undefined;
const stopVectorSource = new OlVectorSource({ features: [] }); const stopVectorSource = new OlVectorSource({ features: [] });
@@ -76,6 +76,7 @@ export const Map: ParentComponent<{}> = () => {
], ],
overlays: [overlay], overlays: [overlay],
}); });
console.log("map=", map);
map.on('singleclick', onClickedMap); map.on('singleclick', onClickedMap);
} }
@@ -108,7 +109,10 @@ export const Map: ParentComponent<{}> = () => {
} }
} }
onMount(() => buildMap(mapDiv)); onMount(() => {
buildMap(mapDiv);
})
;
// Filling the map with stops shape // Filling the map with stops shape
createEffect(() => { createEffect(() => {
@@ -116,8 +120,10 @@ export const Map: ParentComponent<{}> = () => {
const foundStopIds = new Set(); const foundStopIds = new Set();
for (const foundStop of stops) { for (const foundStop of stops) {
foundStopIds.add(foundStop.id); foundStopIds.add(foundStop.id);
if (foundStop.stops !== undefined) {
foundStop.stops.forEach(s => foundStopIds.add(s.id)); foundStop.stops.forEach(s => foundStopIds.add(s.id));
} }
}
for (const [stopIdStr, feature] of Object.entries(displayedFeatures)) { for (const [stopIdStr, feature] of Object.entries(displayedFeatures)) {
const stopId = parseInt(stopIdStr); const stopId = parseInt(stopIdStr);
@@ -205,8 +211,8 @@ export const Map: ParentComponent<{}> = () => {
} }
return <> return <>
<div ref={mapDiv} class="map"> <div ref={mapDiv!} class="map">
<StopPopup ref={popup} stop={selectedMapStop()} show={isPopupDisplayed()} /> <StopPopup ref={popup!} stop={selectedMapStop()} show={isPopupDisplayed()} />
</div> </div>
<For each={getFoundStops()}>{(stop) => <MapStop stop={stop} selected={selectedMapStop()} />}</For> <For each={getFoundStops()}>{(stop) => <MapStop stop={stop} selected={selectedMapStop()} />}</For>
</>; </>;

View File

@@ -149,8 +149,7 @@ export function SearchProvider(props: { children: JSX.Element }) {
setStore('mapFeatures', stopId, feature); setStore('mapFeatures', stopId, feature);
}; };
return ( return <SearchContext.Provider value={{
<SearchContext.Provider value={{
getSearchText, setSearchText, getSearchText, setSearchText,
getFoundStops, setFoundStops, getFoundStops, setFoundStops,
getDisplayedPanelId, setDisplayedPanelId, getDisplayedPanelId, setDisplayedPanelId,
@@ -160,6 +159,5 @@ export function SearchProvider(props: { children: JSX.Element }) {
getMapFeature, getAllMapFeatures, setMapFeature, getMapFeature, getAllMapFeatures, setMapFeature,
}}> }}>
{props.children} {props.children}
</SearchContext.Provider> </SearchContext.Provider>;
);
} }

View File

@@ -33,8 +33,7 @@ const StopRepr: VoidComponent<{ stop: Stop }> = (props) => {
const [lineReprs] = createResource<JSX.Element[], string[]>(props.stop.lines, fetchLinesRepr); const [lineReprs] = createResource<JSX.Element[], string[]>(props.stop.lines, fetchLinesRepr);
return ( return <div class="stop">
<div class="stop">
<svg class="name" viewBox={`0 0 215 ${fontSize}`}> <svg class="name" viewBox={`0 0 215 ${fontSize}`}>
<text <text
x="100%" y="55%" x="100%" y="55%"
@@ -44,8 +43,7 @@ const StopRepr: VoidComponent<{ stop: Stop }> = (props) => {
</text> </text>
</svg> </svg>
<For each={lineReprs()}>{(line: JSX.Element) => line}</For> <For each={lineReprs()}>{(line: JSX.Element) => line}</For>
</div> </div>;
);
} }
@@ -91,12 +89,11 @@ const StopAreaRepr: VoidComponent<{ stop: Stop }> = (props) => {
} }
} }
const sortedTransportModes = Object.keys(byModeReprs).sort((x, y) => TransportModeWeights[x] < const sortedTransportModes = Object.keys(byModeReprs).sort((x, y) => TransportModeWeights[x] < TransportModeWeights[y] ? 1 : -1);
TransportModeWeights[y] ? 1 : -1);
return ( return <div class="lineRepr">
<div class="lineRepr"> <For each={sortedTransportModes}>
<For each={sortedTransportModes}>{(transportMode) => { {(transportMode) => {
const reprs = byModeReprs[transportMode]; const reprs = byModeReprs[transportMode];
const lineNames = Object.keys(reprs.lines).sort((x, y) => x.localeCompare(y)); const lineNames = Object.keys(reprs.lines).sort((x, y) => x.localeCompare(y));
return <> return <>
@@ -104,16 +101,14 @@ const StopAreaRepr: VoidComponent<{ stop: Stop }> = (props) => {
<div class="linesRepresentationMatrix"> <div class="linesRepresentationMatrix">
<For each={lineNames}>{(lineName) => reprs.lines[lineName]}</For> <For each={lineNames}>{(lineName) => reprs.lines[lineName]}</For>
</div> </div>
</> </>;
}} }}</For>
</For> </div>;
</div >
);
} }
const [lineReprs] = createResource(props.stop, fetchLinesRepr); const [lineReprs] = createResource(props.stop, fetchLinesRepr);
return ( return <div
<div
class="stop" class="stop"
onClick={() => setDisplayedStops([props.stop])} onClick={() => setDisplayedStops([props.stop])}
onMouseEnter={() => setHighlightedStop(props.stop)} onMouseEnter={() => setHighlightedStop(props.stop)}
@@ -123,21 +118,17 @@ const StopAreaRepr: VoidComponent<{ stop: Stop }> = (props) => {
<ScrollingText height={fontSize} width={100} content={props.stop.name} /> <ScrollingText height={fontSize} width={100} content={props.stop.name} />
</div> </div>
{lineReprs()} {lineReprs()}
</div> </div>;
);
} }
export const StopsPanel: ParentComponent<{ stops: Stop[], show: boolean }> = (props) => { export const StopsPanel: ParentComponent<{ stops: Stop[], show: boolean }> = (props) => {
return ( return <div classList={{ "stopPanel": true, "displayed": props.show }}>
<div classList={{ "stopPanel": true, "displayed": props.show }}>
<For each={props.stops.sort((x, y) => x.name.localeCompare(y.name))}> <For each={props.stops.sort((x, y) => x.name.localeCompare(y.name))}>
{(stop) => { {(stop) => {
return <Show when={stop.stops !== undefined} fallback={<StopRepr stop={stop} />}> return <Show when={stop.stops !== undefined} fallback={<StopRepr stop={stop} />}>
<StopAreaRepr stop={stop} /> <StopAreaRepr stop={stop} />
</Show>; </Show>;
}} }}</For>
</For> </div>;
</div>
);
} }

View File

@@ -29,8 +29,8 @@ export const StopPopup: ParentComponent<{ stop: Stop, show: boolean }> = (props)
} }
const [destinations] = createResource(() => props.stop, getDestinations); const [destinations] = createResource(() => props.stop, getDestinations);
return (
<div ref={popupDiv} classList={{ "popup": true, "displayed": props.show }}> return <div ref={popupDiv} classList={{ "popup": true, "displayed": props.show }}>
<div class="header">{props.stop?.name}</div> <div class="header">{props.stop?.name}</div>
<div class="body" > <div class="body" >
<For each={destinations()}> <For each={destinations()}>
@@ -41,9 +41,7 @@ export const StopPopup: ParentComponent<{ stop: Stop, show: boolean }> = (props)
<ScrollingText height={10} width={130} content={dst.destinations.join('/')} /> <ScrollingText height={10} width={130} content={dst.destinations.join('/')} />
</div> </div>
</div>; </div>;
}} }}</For>
</For>
</div> </div>
</div > </div>;
);
} }

View File

@@ -33,6 +33,7 @@
background-color: transparent; background-color: transparent;
.leftAddon { .leftAddon {
width: 17%; width: 17%;

View File

@@ -1,4 +1,4 @@
import { createEffect, For, JSX, lazy, ParentComponent, useContext, Show, VoidComponent } from 'solid-js'; import { createEffect, For, JSX, ParentComponent, useContext, Show, VoidComponent } from 'solid-js';
import { lazily } from 'solidjs-lazily'; import { lazily } from 'solidjs-lazily';
import { createScrollPosition } from "@solid-primitives/scroll"; import { createScrollPosition } from "@solid-primitives/scroll";
@@ -16,11 +16,10 @@ import "./stopsSearchMenu.scss";
const StopNameInput: VoidComponent<{ onInput: JSX.EventHandler<HTMLInputElement, InputEvent>, leftAddon: string, placeholder: string }> = (props) => { const StopNameInput: VoidComponent<{ onInput: JSX.EventHandler<HTMLInputElement, InputEvent>, leftAddon: string, placeholder: string }> = (props) => {
return ( return <div class="stopNameInput">
<div class="stopNameInput">
<div class="leftAddon">{props.leftAddon}</div> <div class="leftAddon">{props.leftAddon}</div>
<input type="text" oninput={props.onInput} placeholder={props.placeholder} /> <input type="text" oninput={props.onInput} placeholder={props.placeholder} />
</div>); </div>;
}; };
@@ -40,8 +39,7 @@ const Header: VoidComponent<{ title: string, minCharsNb: number }> = (props) =>
} }
} }
return ( return <div class="header">
<div class="header">
<div class="title"> <div class="title">
<svg viewBox="0 0 1260 50"> <svg viewBox="0 0 1260 50">
<text x="0" y="50%" dominant-baseline="middle" font-size="50" style="fill: #ffffff"> <text x="0" y="50%" dominant-baseline="middle" font-size="50" style="fill: #ffffff">
@@ -50,8 +48,7 @@ const Header: VoidComponent<{ title: string, minCharsNb: number }> = (props) =>
</svg> </svg>
</div> </div>
<StopNameInput onInput={onStopNameInput} leftAddon="🚉 🚏" placeholder="Stop name..." /> <StopNameInput onInput={onStopNameInput} leftAddon="🚉 🚏" placeholder="Stop name..." />
</div > </div>;
);
}; };
@@ -71,13 +68,15 @@ const StopsPanels: ParentComponent<{ maxStopsPerPanel: number }> = (props) => {
yStopsPanelsScroll(); yStopsPanelsScroll();
for (const panel of getPanels()) { for (const panel of getPanels()) {
const panelDiv = panel.panel(); const panelDiv = panel.panel;
if (panelDiv != null) {
const panelDivClientRect = panelDiv.getBoundingClientRect(); const panelDivClientRect = panelDiv.getBoundingClientRect();
if (panelDivClientRect.y > 0) { if (panelDivClientRect.y > 0) {
setDisplayedPanelId(panel.position); setDisplayedPanelId(panel.position);
break; break;
} }
} }
}
}); });
return ( return (
@@ -133,7 +132,8 @@ const MapPlaceholder: VoidComponent<{}> = () => {
} }
return <div return <div
class="mapPlaceholder" ondblclick={() => onDoubleClick()}> class="mapPlaceholder"
ondblclick={() => onDoubleClick()}>
Double-clic pour activer la carte Double-clic pour activer la carte
</div>; </div>;
}; };
@@ -166,35 +166,31 @@ const Footer: VoidComponent<{}> = () => {
const { getDisplayedPanelId, getPanels } = searchStore; const { getDisplayedPanelId, getPanels } = searchStore;
return ( return <div class="footer">
<div class="footer">
<For each={getPanels()}> <For each={getPanels()}>
{(panel) => { {(panel) => {
const position = panel.position; const position = panel.position;
return ( return (
<div> <div>
<svg viewBox="0 0 29 29"> <svg viewBox="0 0 29 29">
<circle cx="50%" cy="50%" r="13" stroke="#ffffff" stroke-width="3" <circle
cx="50%" cy="50%" r="13" stroke="#ffffff" stroke-width="3"
style={{ fill: `var(--idfm-${position == getDisplayedPanelId() ? "white" : "black"})` }} style={{ fill: `var(--idfm-${position == getDisplayedPanelId() ? "white" : "black"})` }}
/> />
</svg> </svg>
</div> </div>
); );
}} }}</For>
</For> </div>;
</div>
);
}; };
export const StopsSearchMenu: VoidComponent = () => { export const StopsSearchMenu: VoidComponent = () => {
return ( return <div class="stopSearchMenu">
<div class="stopSearchMenu">
<SearchProvider> <SearchProvider>
<Header title="Recherche de l'arrêt..." minCharsNb={4} /> <Header title="Recherche de l'arrêt..." minCharsNb={4} />
<Body /> <Body />
<Footer /> <Footer />
</SearchProvider> </SearchProvider>
</div> </div>;
);
}; };

View File

@@ -8,7 +8,7 @@ export enum TrafficStatus {
export class Passage { export class Passage {
line: number; line: number;
operator: string; operator: number;
destinations: string[]; destinations: string[];
atStop: boolean; atStop: boolean;
aimedArrivalTs: number; aimedArrivalTs: number;
@@ -19,7 +19,7 @@ export class Passage {
arrivalStatus: string; arrivalStatus: string;
departStatus: string; departStatus: string;
constructor(line: number, operator: string, destinations: string[], atStop: boolean, aimedArrivalTs: number, constructor(line: number, operator: number, destinations: string[], atStop: boolean, aimedArrivalTs: number,
expectedArrivalTs: number, arrivalPlatformName: string, aimedDepartTs: number, expectedDepartTs: number, expectedArrivalTs: number, arrivalPlatformName: string, aimedDepartTs: number, expectedDepartTs: number,
arrivalStatus: string, departStatus: string) { arrivalStatus: string, departStatus: string) {
this.line = line; this.line = line;
@@ -34,7 +34,7 @@ export class Passage {
this.arrivalStatus = arrivalStatus; this.arrivalStatus = arrivalStatus;
this.departStatus = departStatus; this.departStatus = departStatus;
} }
}; }
export type Passages = Record<string, Record<string, Passage[]>>; export type Passages = Record<string, Record<string, Passage[]>>;
@@ -45,9 +45,9 @@ export class Stop {
epsg3857_x: number; epsg3857_x: number;
epsg3857_y: number; epsg3857_y: number;
stops: Stop[]; stops: Stop[];
lines: string[]; lines: number[];
constructor(id: number, name: string, town: string, epsg3857_x: number, epsg3857_y: number, stops: Stop[], lines: string[]) { constructor(id: number, name: string, town: string, epsg3857_x: number, epsg3857_y: number, stops: Stop[], lines: number[]) {
this.id = id; this.id = id;
this.name = name; this.name = name;
this.town = town; this.town = town;
@@ -59,7 +59,7 @@ export class Stop {
this.lines.push(...stop.lines); this.lines.push(...stop.lines);
} }
} }
}; }
export type Stops = Record<number, Stop>; export type Stops = Record<number, Stop>;
@@ -77,12 +77,12 @@ export class StopShape {
this.epsg3857_bbox = epsg3857_bbox; this.epsg3857_bbox = epsg3857_bbox;
this.epsg3857_points = epsg3857_points; this.epsg3857_points = epsg3857_points;
} }
}; }
export type StopShapes = Record<number, StopShape>; export type StopShapes = Record<number, StopShape>;
export class Line { export class Line {
id: string; id: number;
shortName: string; shortName: string;
name: string; name: string;
status: string; // TODO: Use an enum status: string; // TODO: Use an enum
@@ -95,7 +95,7 @@ export class Line {
audibleSignsAvailable: string; // TODO: Use an enum audibleSignsAvailable: string; // TODO: Use an enum
stopIds: number[]; stopIds: number[];
constructor(id: string, shortName: string, name: string, status: string, transportMode: string, backColorHexa: string, constructor(id: number, shortName: string, name: string, status: string, transportMode: string, backColorHexa: string,
foreColorHexa: string, operatorId: number, accessibility: boolean, visualSignsAvailable: string, foreColorHexa: string, operatorId: number, accessibility: boolean, visualSignsAvailable: string,
audibleSignsAvailable: string, stopIds: number[]) { audibleSignsAvailable: string, stopIds: number[]) {
this.id = id; this.id = id;
@@ -111,6 +111,6 @@ export class Line {
this.audibleSignsAvailable = audibleSignsAvailable; this.audibleSignsAvailable = audibleSignsAvailable;
this.stopIds = stopIds; this.stopIds = stopIds;
} }
}; }
export type Lines = Record<string, Line>; export type Lines = Record<string, Line>;

View File

@@ -26,7 +26,7 @@ export const TransportModeWeights: Record<string, number> = {
export function getTransportModeSrc(mode: string, color: boolean = true): string | undefined { export function getTransportModeSrc(mode: string, color: boolean = true): string | undefined {
let ret = undefined; let ret = undefined;
if (validTransportModes.includes(mode)) { if (validTransportModes.includes(mode)) {
return `/carrramba-encore-rate/public/symbole_${mode}_${color ? "" : "support_fonce_"}RVB.svg`; return `/symbole_${mode}_${color ? "" : "support_fonce_"}RVB.svg`;
} }
return ret; return ret;
} }
@@ -36,11 +36,11 @@ export function renderLineTransportMode(line: Line): JSX.Element {
} }
function renderBusLinePicto(line: Line): JSX.Element { function renderBusLinePicto(line: Line): JSX.Element {
return ( return <div class="busLinePicto">
<div class="busLinePicto">
<svg viewBox="0 0 31.5 14"> <svg viewBox="0 0 31.5 14">
<rect x="0" y="0" width="31.5" height="14" rx="1.5" ry="1.5" style={{ fill: `#${line.backColorHexa}` }} /> <rect x="0" y="0" width="31.5" height="14" rx="1.5" ry="1.5" style={{ fill: `#${line.backColorHexa}` }} />
<text x="50%" <text
x="50%"
y="55%" y="55%"
dominant-baseline="middle" dominant-baseline="middle"
text-anchor="middle" text-anchor="middle"
@@ -49,18 +49,17 @@ function renderBusLinePicto(line: Line): JSX.Element {
{line.shortName} {line.shortName}
</text> </text>
</svg> </svg>
</div> </div>;
);
} }
function renderTramLinePicto(line: Line): JSX.Element { function renderTramLinePicto(line: Line): JSX.Element {
const lineStyle = { fill: `#${line.backColorHexa}` }; const lineStyle = { fill: `#${line.backColorHexa}` };
return ( return <div class="tramLinePicto">
<div class="tramLinePicto">
<svg viewBox="0 0 20 20"> <svg viewBox="0 0 20 20">
<rect x="0" y="0" width="20" height="3" rx="1" ry="1" style={lineStyle} /> <rect x="0" y="0" width="20" height="3" rx="1" ry="1" style={lineStyle} />
<rect x="0" y="17" width="20" height="3" rx="1" ry="1" style={lineStyle} /> <rect x="0" y="17" width="20" height="3" rx="1" ry="1" style={lineStyle} />
<text x="50%" <text
x="50%"
y="55%" y="55%"
dominant-baseline="middle" dominant-baseline="middle"
text-anchor="middle" text-anchor="middle"
@@ -69,16 +68,15 @@ function renderTramLinePicto(line: Line): JSX.Element {
{line.shortName} {line.shortName}
</text> </text>
</svg> </svg>
</div> </div>;
);
} }
function renderMetroLinePicto(line: Line): JSX.Element { function renderMetroLinePicto(line: Line): JSX.Element {
return ( return <div class="metroLinePicto">
<div class="metroLinePicto">
<svg viewBox="0 0 20 20"> <svg viewBox="0 0 20 20">
<circle cx="10" cy="10" r="10" style={{ fill: `#${line.backColorHexa}` }} /> <circle cx="10" cy="10" r="10" style={{ fill: `#${line.backColorHexa}` }} />
<text x="50%" <text
x="50%"
y="55%" y="55%"
dominant-baseline="middle" dominant-baseline="middle"
text-anchor="middle" text-anchor="middle"
@@ -86,16 +84,15 @@ function renderMetroLinePicto(line: Line): JSX.Element {
{line.shortName} {line.shortName}
</text> </text>
</svg> </svg>
</div> </div>;
);
} }
function renderTrainLinePicto(line: Line): JSX.Element { function renderTrainLinePicto(line: Line): JSX.Element {
return ( return <div class="trainLinePicto">
<div class="trainLinePicto">
<svg viewBox="0 0 20 20"> <svg viewBox="0 0 20 20">
<rect x="0" y="0" width="20" height="20" rx="4.5" ry="4.5" style={{ fill: `#${line.backColorHexa}` }} /> <rect x="0" y="0" width="20" height="20" rx="4.5" ry="4.5" style={{ fill: `#${line.backColorHexa}` }} />
<text x="50%" <text
x="50%"
y="55%" y="55%"
dominant-baseline="middle" dominant-baseline="middle"
text-anchor="middle" text-anchor="middle"
@@ -104,8 +101,7 @@ function renderTrainLinePicto(line: Line): JSX.Element {
{line.shortName} {line.shortName}
</text> </text>
</svg> </svg>
</div> </div>;
);
} }
export function renderLinePicto(line: Line): JSX.Element { export function renderLinePicto(line: Line): JSX.Element {
@@ -152,8 +148,7 @@ export const ScrollingText: VoidComponent<{ height: number, width: number, conte
} }
}); });
return ( return <svg ref={viewBoxRef} viewBox={`0 0 ${props.width} ${props.height}`}>
<svg ref={viewBoxRef} viewBox={`0 0 ${props.width} ${props.height}`}>
<text <text
ref={textRef} ref={textRef}
x="0%" y="55%" x="0%" y="55%"
@@ -161,6 +156,5 @@ export const ScrollingText: VoidComponent<{ height: number, width: number, conte
font-size={`${props.height}px`}> font-size={`${props.height}px`}>
{props.content} {props.content}
</text> </text>
</svg > </svg>;
);
} }

View File

@@ -4,6 +4,7 @@
"jsxImportSource": "solid-js", "jsxImportSource": "solid-js",
"noImplicitAny": true, "noImplicitAny": true,
"target": "ES6", "target": "ES6",
"module": "esnext",
"moduleResolution": "node", "moduleResolution": "node",
"allowJs": true, "allowJs": true,
"outDir": "build", "outDir": "build",

BIN
medias/presentation.mp4 (Stored with Git LFS) Normal file

Binary file not shown.

BIN
medias/presentation.png (Stored with Git LFS) Normal file

Binary file not shown.