diff --git a/angular.json b/angular.json index 93dbf3b..dc611de 100644 --- a/angular.json +++ b/angular.json @@ -33,6 +33,7 @@ "styles": [ "node_modules/primeng/resources/themes/lara-light-blue/theme.css", "node_modules/primeng/resources/primeng.min.css", + "node_modules/maplibre-gl/dist/maplibre-gl.css", "src/styles.scss" ], "stylePreprocessorOptions": { diff --git a/package.json b/package.json index b7ad44b..b1e8a88 100644 --- a/package.json +++ b/package.json @@ -35,11 +35,14 @@ "@ngrx/store": "^18.0.2", "@ngrx/store-devtools": "^18.0.1", "@planess/train-a-backend": "^0.0.3", + "@types/geojson": "^7946.0.14", "@types/jest": "^29.5.12", + "@types/maplibre-gl": "^1.14.0", "eslint-plugin-unused-imports": "^4.1.3", "express": "^4.18.2", "jest": "^29.7.0", "jest-preset-angular": "^14.2.2", + "maplibre-gl": "^4.5.2", "modern-normalize": "^3.0.0", "primeicons": "^7.0.0", "primeng": "^17.18.9", diff --git a/public/styles/common.scss b/public/styles/common.scss index cea2d00..6565ad2 100644 --- a/public/styles/common.scss +++ b/public/styles/common.scss @@ -23,6 +23,14 @@ body { body { margin: 0; + font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Fira Sans', + 'Droid Sans', 'Helvetica Neue', sans-serif; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; +} + +code { + font-family: source-code-pro, Menlo, Monaco, Consolas, 'Courier New', monospace; } .hidden { diff --git a/src/app/admin/components/create-station-form/create-station-form.component.html b/src/app/admin/components/create-station-form/create-station-form.component.html new file mode 100644 index 0000000..2aeaccf --- /dev/null +++ b/src/app/admin/components/create-station-form/create-station-form.component.html @@ -0,0 +1,80 @@ +
+

Create Station

+
+ + + + + + + +
+ + diff --git a/src/app/admin/components/create-station-form/create-station-form.component.scss b/src/app/admin/components/create-station-form/create-station-form.component.scss new file mode 100644 index 0000000..d15a6de --- /dev/null +++ b/src/app/admin/components/create-station-form/create-station-form.component.scss @@ -0,0 +1,27 @@ +.form { + display: flex; + flex-direction: column; + gap: 1rem; + margin-top: 4.813rem; +} + +.controls { + display: flex; + flex-direction: column; + gap: 1rem; +} + +.control { + width: 100%; + + &-label { + display: flex; + flex-direction: column; + gap: 0.5rem; + } +} + +.submit { + gap: 0.5rem; + justify-content: center; +} diff --git a/src/app/admin/components/create-station-form/create-station-form.component.ts b/src/app/admin/components/create-station-form/create-station-form.component.ts new file mode 100644 index 0000000..b75a4ad --- /dev/null +++ b/src/app/admin/components/create-station-form/create-station-form.component.ts @@ -0,0 +1,107 @@ +import { ChangeDetectionStrategy, Component, inject, OnDestroy, OnInit, signal } from '@angular/core'; +import { FormBuilder, ReactiveFormsModule, Validators } from '@angular/forms'; + +import { MessageService } from 'primeng/api'; +import { ButtonModule } from 'primeng/button'; +import { FloatLabelModule } from 'primeng/floatlabel'; +import { InputNumberModule } from 'primeng/inputnumber'; +import { RippleModule } from 'primeng/ripple'; +import { ToastModule } from 'primeng/toast'; +import { map, Observable, of, Subscription, switchMap } from 'rxjs'; + +import { StationsService } from '@/app/api/stationsService/stations.service'; +import MESSAGE_STATUS from '@/app/shared/constants/message-status'; + +import { MapService } from '../../services/map/map.service'; + +@Component({ + selector: 'app-create-station-form', + standalone: true, + imports: [FloatLabelModule, InputNumberModule, ReactiveFormsModule, RippleModule, ButtonModule, ToastModule], + providers: [MessageService], + templateUrl: './create-station-form.component.html', + styleUrl: './create-station-form.component.scss', + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class CreateStationFormComponent implements OnInit, OnDestroy { + private fb = inject(FormBuilder); + private mapService = inject(MapService); + private stationsService = inject(StationsService); + private messageService = inject(MessageService); + private subscription = new Subscription(); + + public isStationCreated = signal(false); + + private createStation(city: string, latitude: number, longitude: number): Observable<{ id: number }> { + return this.stationsService.createNewStation({ + city, + latitude, + longitude, + relations: [], + }); + } + + private createMarker(city: string, latitude: number, longitude: number): void { + this.mapService.createNewMarker({ + city, + lat: latitude, + lng: longitude, + }); + } + + public createStationForm = this.fb.nonNullable.group({ + city: ['', [Validators.required.bind(this)]], + latitude: [0, [Validators.required.bind(this)]], + longitude: [0, [Validators.required.bind(this)]], + }); + + public submitForm(): void { + this.createStationForm.markAllAsTouched(); + this.createStationForm.updateValueAndValidity(); + + if (this.createStationForm.valid) { + this.isStationCreated.set(true); + const { city, latitude, longitude } = this.createStationForm.getRawValue(); + this.subscription.add( + this.stationsService + .isStationInCity(city) + .pipe( + switchMap((exists) => + exists + ? of(null) + : this.createStation(city, latitude, longitude).pipe( + map((id) => { + this.isStationCreated.set(false); + this.createStationForm.reset(); + this.messageService.add({ + severity: MESSAGE_STATUS.SUCCESS, + summary: 'Success!', + detail: `Station created with id: ${id.id}`, + }); + }), + ), + ), + map((exists) => (exists ? of(null) : this.createMarker(city, latitude, longitude))), + ) + .subscribe({ + error: (error: Error) => { + this.isStationCreated.set(false); + this.messageService.add({ severity: MESSAGE_STATUS.ERROR, summary: error.name, detail: error.message }); + }, + }), + ); + } + } + + public ngOnInit(): void { + this.subscription.add( + this.mapService.getLngLat().subscribe((lngLat) => { + this.createStationForm.patchValue({ longitude: lngLat.lng, latitude: lngLat.lat }); + }), + ); + } + + public ngOnDestroy(): void { + this.subscription.unsubscribe(); + } +} diff --git a/src/app/admin/components/map/map.component.html b/src/app/admin/components/map/map.component.html new file mode 100644 index 0000000..7af8fac --- /dev/null +++ b/src/app/admin/components/map/map.component.html @@ -0,0 +1,5 @@ +
+
+ +
+
diff --git a/src/app/admin/components/map/map.component.scss b/src/app/admin/components/map/map.component.scss new file mode 100644 index 0000000..a8820d6 --- /dev/null +++ b/src/app/admin/components/map/map.component.scss @@ -0,0 +1,16 @@ +:host { + display: flex; +} + +.map-wrap { + position: relative; + width: 100%; + height: calc(100vh - 4.813rem); + margin-top: 4.813rem; +} + +.map { + position: absolute; + width: 100%; + height: 100%; +} diff --git a/src/app/admin/components/map/map.component.ts b/src/app/admin/components/map/map.component.ts new file mode 100644 index 0000000..167f069 --- /dev/null +++ b/src/app/admin/components/map/map.component.ts @@ -0,0 +1,83 @@ +import { + AfterViewInit, + ChangeDetectionStrategy, + Component, + ElementRef, + inject, + OnDestroy, + OnInit, + signal, + ViewChild, +} from '@angular/core'; + +import { Map, NavigationControl } from 'maplibre-gl'; +import { SkeletonModule } from 'primeng/skeleton'; +import { Subscription } from 'rxjs'; + +import ENVIRONMENTS from '@/environment/environment'; + +import { INITIAL_MAP_STATE } from '../../constants/initial-map-state'; +import { MapService } from '../../services/map/map.service'; + +@Component({ + selector: 'app-map', + standalone: true, + imports: [SkeletonModule], + templateUrl: './map.component.html', + styleUrl: './map.component.scss', + changeDetection: ChangeDetectionStrategy.OnPush, +}) +export class MapComponent implements OnInit, AfterViewInit, OnDestroy { + private mapService = inject(MapService); + private subscription = new Subscription(); + private map!: Map; + + @ViewChild('map') + private mapContainer!: ElementRef; + + public isMapLoaded = signal(false); + + private initMap(): void { + this.map = new Map({ + container: this.mapContainer.nativeElement, + style: `https://api.maptiler.com/maps/streets-v2/style.json?key=${ENVIRONMENTS.MAP_KEY}`, + center: [INITIAL_MAP_STATE.lng, INITIAL_MAP_STATE.lat], + zoom: INITIAL_MAP_STATE.zoom, + }); + + this.map.addControl(new NavigationControl({})); + } + + private initMapClickHandler(): void { + this.map.on('click', ({ lngLat }) => { + this.mapService.setLngLat({ lng: lngLat.lng, lat: lngLat.lat }); + }); + + this.map.on('load', () => { + this.isMapLoaded.set(true); + }); + } + + public ngOnInit(): void { + this.subscription.add( + this.mapService.getNewMarker().subscribe((marker) => { + marker.addTo(this.map); + marker.togglePopup(); + this.map.flyTo({ + center: [marker.getLngLat().lng, marker.getLngLat().lat], + zoom: INITIAL_MAP_STATE.zoom, + }); + }), + ); + } + + public ngAfterViewInit(): void { + this.initMap(); + this.initMapClickHandler(); + } + + public ngOnDestroy(): void { + this.map.remove(); + this.subscription.unsubscribe(); + } +} diff --git a/src/app/admin/components/sidebar/sidebar.component.scss b/src/app/admin/components/sidebar/sidebar.component.scss index 71acb3d..9c2f644 100644 --- a/src/app/admin/components/sidebar/sidebar.component.scss +++ b/src/app/admin/components/sidebar/sidebar.component.scss @@ -5,12 +5,28 @@ height: 100%; background-color: $gray-200; border-right: $one solid $gray-100; + + ::ng-deep .p-element { + display: flex; + align-items: center; + justify-content: center; + + width: $offset-ms; + height: $offset-ms; + padding: 0; + + font-size: $font-size-xs; + + &.link-back { + margin: $offset-xxs; + margin-right: 0; + } + } } .links { display: flex; flex-direction: column; - height: 100%; } .link { @@ -38,20 +54,3 @@ background-color: $gray-100; } } - -::ng-deep .p-element { - display: flex; - align-items: center; - justify-content: center; - - width: $offset-ms; - height: $offset-ms; - padding: 0; - - font-size: $font-size-xs; - - &.link-back { - margin: $offset-xxs; - margin-right: 0; - } -} diff --git a/src/app/admin/constants/initial-map-state.ts b/src/app/admin/constants/initial-map-state.ts new file mode 100644 index 0000000..52fe797 --- /dev/null +++ b/src/app/admin/constants/initial-map-state.ts @@ -0,0 +1,8 @@ +export const INITIAL_MAP_STATE = { lng: 2.349014, lat: 48.864716, zoom: 10 }; +export const MARKER_PARAMS = { + height: 40, + radius: 40, + offset: 20, + max_width: '5rem', + color: '#3B82F6', +}; diff --git a/src/app/admin/layout/admin-layout/admin-layout.component.scss b/src/app/admin/layout/admin-layout/admin-layout.component.scss index 82c0881..6d11024 100644 --- a/src/app/admin/layout/admin-layout/admin-layout.component.scss +++ b/src/app/admin/layout/admin-layout/admin-layout.component.scss @@ -5,3 +5,7 @@ grid-template-columns: calc($offset-xxxl * 2.6) 1fr; height: 100vh; } + +.content { + padding: 0 $offset-s; +} diff --git a/src/app/admin/pages/stations/stations.component.html b/src/app/admin/pages/stations/stations.component.html index 4fded1c..303cf8b 100644 --- a/src/app/admin/pages/stations/stations.component.html +++ b/src/app/admin/pages/stations/stations.component.html @@ -1 +1,4 @@ -

stations works!

+
+ + +
diff --git a/src/app/admin/pages/stations/stations.component.scss b/src/app/admin/pages/stations/stations.component.scss index e69de29..19857da 100644 --- a/src/app/admin/pages/stations/stations.component.scss +++ b/src/app/admin/pages/stations/stations.component.scss @@ -0,0 +1,7 @@ +@import 'variables'; + +.wrapper { + display: grid; + grid-template-columns: 1fr 25rem; + gap: $offset-s; +} diff --git a/src/app/admin/pages/stations/stations.component.spec.ts b/src/app/admin/pages/stations/stations.component.spec.ts deleted file mode 100644 index c4c2ab4..0000000 --- a/src/app/admin/pages/stations/stations.component.spec.ts +++ /dev/null @@ -1,22 +0,0 @@ -import { ComponentFixture, TestBed } from '@angular/core/testing'; - -import { StationsComponent } from './stations.component'; - -describe('StationsComponent', () => { - let component: StationsComponent; - let fixture: ComponentFixture; - - beforeEach(async () => { - await TestBed.configureTestingModule({ - imports: [StationsComponent], - }).compileComponents(); - - fixture = TestBed.createComponent(StationsComponent); - component = fixture.componentInstance; - fixture.detectChanges(); - }); - - it('should create', () => { - expect(component).toBeTruthy(); - }); -}); diff --git a/src/app/admin/pages/stations/stations.component.ts b/src/app/admin/pages/stations/stations.component.ts index dbfbcbe..1fe9baa 100644 --- a/src/app/admin/pages/stations/stations.component.ts +++ b/src/app/admin/pages/stations/stations.component.ts @@ -1,9 +1,12 @@ import { ChangeDetectionStrategy, Component } from '@angular/core'; +import { CreateStationFormComponent } from '../../components/create-station-form/create-station-form.component'; +import { MapComponent } from '../../components/map/map.component'; + @Component({ selector: 'app-stations', standalone: true, - imports: [], + imports: [MapComponent, CreateStationFormComponent], templateUrl: './stations.component.html', styleUrl: './stations.component.scss', changeDetection: ChangeDetectionStrategy.OnPush, diff --git a/src/app/admin/services/.gitkeep b/src/app/admin/services/.gitkeep deleted file mode 100644 index e69de29..0000000 diff --git a/src/app/admin/services/map/map.service.ts b/src/app/admin/services/map/map.service.ts new file mode 100644 index 0000000..ea014e2 --- /dev/null +++ b/src/app/admin/services/map/map.service.ts @@ -0,0 +1,43 @@ +import { Injectable } from '@angular/core'; + +import { Marker, Popup } from 'maplibre-gl'; +import { BehaviorSubject, Subject } from 'rxjs'; + +import makeFirstLetterToUppercase from '@/app/shared/utils/makeFirstLetterToUppercase'; + +import { MARKER_PARAMS } from '../../constants/initial-map-state'; +import createNewPopupOffsets from '../../utils/createNewPopupOffsets'; + +@Injectable({ + providedIn: 'root', +}) +export class MapService { + private lngLat = new BehaviorSubject({ lng: 0, lat: 0 }); + private newMarker = new Subject(); + + public createNewMarker({ city, lng, lat }: { city: string; lng: number; lat: number }): void { + this.newMarker.next( + new Marker({ color: MARKER_PARAMS.color }) + .setLngLat([lng, lat]) + .setPopup( + new Popup({ className: 'map-popup' }) + .setLngLat({ lng, lat }) + .setText(makeFirstLetterToUppercase(city)) + .setOffset(createNewPopupOffsets()) + .setMaxWidth(MARKER_PARAMS.max_width), + ), + ); + } + + public getLngLat(): BehaviorSubject<{ lng: number; lat: number }> { + return this.lngLat; + } + + public setLngLat({ lng, lat }: { lng: number; lat: number }): void { + this.lngLat.next({ lng, lat }); + } + + public getNewMarker(): Subject { + return this.newMarker; + } +} diff --git a/src/app/admin/utils/createNewPopupOffsets.ts b/src/app/admin/utils/createNewPopupOffsets.ts new file mode 100644 index 0000000..6788d66 --- /dev/null +++ b/src/app/admin/utils/createNewPopupOffsets.ts @@ -0,0 +1,23 @@ +import { Offset, Point } from 'maplibre-gl'; + +import { MARKER_PARAMS } from '../constants/initial-map-state'; + +const createNewPopupOffsets = (): Offset => ({ + top: new Point(0, 0), + left: new Point(MARKER_PARAMS.radius, (MARKER_PARAMS.height - MARKER_PARAMS.radius) * -1), + right: new Point(-MARKER_PARAMS.radius, (MARKER_PARAMS.height - MARKER_PARAMS.radius) * -1), + bottom: new Point(0, -MARKER_PARAMS.height), + center: new Point(0, MARKER_PARAMS.height), + 'top-left': new Point(0, 0), + 'top-right': new Point(0, 0), + 'bottom-left': new Point( + MARKER_PARAMS.offset, + (MARKER_PARAMS.height - MARKER_PARAMS.radius + MARKER_PARAMS.offset) * -1, + ), + 'bottom-right': new Point( + -MARKER_PARAMS.offset, + (MARKER_PARAMS.height - MARKER_PARAMS.radius + MARKER_PARAMS.offset) * -1, + ), +}); + +export default createNewPopupOffsets; diff --git a/src/app/api/stationsService/stations.service.ts b/src/app/api/stationsService/stations.service.ts index 6ed2c49..3f8ec43 100644 --- a/src/app/api/stationsService/stations.service.ts +++ b/src/app/api/stationsService/stations.service.ts @@ -1,7 +1,7 @@ import { HttpClient } from '@angular/common/http'; import { inject, Injectable } from '@angular/core'; -import { Observable } from 'rxjs'; +import { map, Observable } from 'rxjs'; import { NewStation, Station } from '../models/stations'; @@ -23,4 +23,12 @@ export class StationsService { public deleteStation(id: number): Observable { return this.httpClient.delete(`${this.STATION_ENDPOINT}/${id}`); } + + public isStationInCity(station: string): Observable { + return this.getStations().pipe( + map((stations) => + stations.some((stationFromList) => stationFromList.city.toLowerCase() === station.toLowerCase()), + ), + ); + } } diff --git a/src/app/shared/constants/message-status.ts b/src/app/shared/constants/message-status.ts new file mode 100644 index 0000000..29ffc06 --- /dev/null +++ b/src/app/shared/constants/message-status.ts @@ -0,0 +1,9 @@ +const MESSAGE_STATUS = { + SUCCESS: 'success', + ERROR: 'error', + INFO: 'info', + WARNING: 'warning', + DEFAULT: 'default', +} as const; + +export default MESSAGE_STATUS; diff --git a/src/app/shared/utils/makeFirstLetterToUppercase.ts b/src/app/shared/utils/makeFirstLetterToUppercase.ts new file mode 100644 index 0000000..55d500e --- /dev/null +++ b/src/app/shared/utils/makeFirstLetterToUppercase.ts @@ -0,0 +1,3 @@ +const makeFirstLetterToUppercase = (text: string): string => text.charAt(0).toUpperCase() + text.slice(1); + +export default makeFirstLetterToUppercase; diff --git a/src/environment/environment.ts b/src/environment/environment.ts new file mode 100644 index 0000000..14f31f1 --- /dev/null +++ b/src/environment/environment.ts @@ -0,0 +1,5 @@ +const ENVIRONMENTS = { + MAP_KEY: '1bdk2M9O2x8FBX6D1NjO', +} as const; + +export default ENVIRONMENTS; diff --git a/tsconfig.json b/tsconfig.json index 41de90c..da35bd4 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -4,6 +4,8 @@ "outDir": "./dist/out-tsc", "strict": true, "noImplicitOverride": true, + "forceConsistentCasingInFileNames": true, + "downlevelIteration": true, "noPropertyAccessFromIndexSignature": true, "noImplicitReturns": true, "noFallthroughCasesInSwitch": true,