;\n\nexport interface PointWithId {\n id: string;\n locked?: boolean;\n}\n\nexport type DisplayablePointOfInterest = PointOfInterest &\n DisplayableMapPoint &\n PointWithId;\n\nexport enum PointSetDisplayType {\n NoDisplay = 0,\n Markers = 1 << 0,\n Polyline = 1 << 1,\n All = ~(~0 << 2),\n}\nexport class PointSet {\n readonly id: PointSetId = Symbol();\n iconClass: string;\n color: string;\n zIndex = 0;\n polyinfo?: PointSetPolyInfo;\n constructor(\n public points: P[],\n public displayType: PointSetDisplayType,\n iconClass?: string,\n color?: string,\n ) {\n this.iconClass = iconClass ? iconClass : '';\n this.color = color ? color : '';\n }\n}\n\nexport type MapMarkerProps = PointDisplayOptions & { point: MapPoint };\n\n/**\n * Some common categories for points of interest.\n */\nexport enum InterestType {\n ACCOMODATION = 'accomodation',\n BUILDING = 'building',\n CATERING = 'restaurant',\n NATURE = 'nature',\n TOURISM = 'tourism',\n LEISURE = 'leisure',\n}\n","import { computed, ComputedRef } from 'vue';\nimport { DI } from '@/di';\nimport { MapMarkerProps, PointSetDisplayType } from '@/map-display';\nimport { PointSetService } from '@/services/map-service/points/point-set.interface';\n\nexport class PointSetCollapse {\n private readonly service = DI.get(PointSetService);\n\n readonly points: ComputedRef = computed(() =>\n this.service\n .getPointSets()\n .value.filter(\n (set) =>\n (set.displayType & PointSetDisplayType.Markers) ===\n PointSetDisplayType.Markers,\n )\n .flatMap((set) =>\n set.points.map((point) => {\n const display = point.displayOptions;\n return {\n point,\n ...display,\n iconClass: display?.iconClass || set.iconClass,\n color: display?.color || set.color,\n zIndex: display?.zIndex || set.zIndex,\n };\n }),\n ),\n );\n}\n","export const BASE_PIN_COLOR = '#3822a8';\nexport const SEARCH_HIGHLIGHT_PIN_COLOR = '#8295ff';\nexport const CITY_STAY_AT_PIN_COLOR = '#ba7600';\nexport const CITY_STAY_AT_PIN_HIGHLIGHT = '#d08600';\nexport const CITY_VIA_PIN_COLOR = '#e87f2e';\nexport const CITY_VIA_PIN_HIGHLIGHT = '#ff9f30';\n\nexport const CITY_PIN_ICON_CLASS = 'fas fa-city';\nexport const SEARCH_PIN_ICON_CLASS = 'fas fa-search';\nexport const GENERIC_TRIP_POINT_ICON_CLASS = 'fas fa-map-pin';\n\nexport const CITY_PIN_Z_INDEX = 100;\nexport const SEARCH_PIN_Z_INDEX = 200;\n\nexport const BASE_PATH_COLOR = 'black';\nexport const IN_CITY_PATH_COLOR = '#8b5fc9';\nexport const CITY_VIA_PATH_COLOR = '#ff8969';\n\nexport const CITY_DAY_COLORS = [\n '#B01515',\n '#D4A200',\n '#087A00',\n '#17BD8B',\n '#00DDFF',\n '#1522D6',\n '#801094',\n '#E01991',\n];\n\nexport const CITY_DAY_HIGHLIGHTS = [\n '#D03535',\n '#F4C230',\n '#28BB30',\n '#37DDAB',\n '#77FFFF',\n '#4562FF',\n '#A030C8',\n '#FF69C4',\n];\n","import { DI } from '@/di';\nimport { PointSetService } from '@/services/map-service/points/point-set.interface';\nimport { MapPoint, PointSetDisplayType } from '@/map-display';\nimport { computed, ComputedRef, DeepReadonly } from 'vue';\nimport { BASE_PATH_COLOR } from '@/icon-styling';\nimport { ComputedPolylineManager } from '@/services/map-service/points/computed-polylines.interface';\n\ntype ComputedPoly = DeepReadonly<{\n points: MapPoint[];\n color?: string;\n dashPattern?: number[];\n}>;\nexport class PolylineFinder {\n private readonly setService = DI.get(PointSetService);\n private readonly polyManager = DI.get(\n ComputedPolylineManager,\n );\n\n private buildOptions(\n color?: string,\n dashArray?: string,\n dashOffset?: string,\n ) {\n return {\n color: color && color.length > 0 ? color : BASE_PATH_COLOR,\n dashArray,\n dashOffset: dashOffset || (dashArray ? '0' : undefined),\n };\n }\n\n readonly lines: ComputedRef = computed(() => {\n const pointSetPolys: ComputedPoly[] = this.setService\n .getPointSets()\n .value.filter(\n (s) =>\n (s.displayType & PointSetDisplayType.Polyline) ===\n PointSetDisplayType.Polyline,\n )\n .map((s) => {\n const { color, points } = s;\n const polyColor = s.polyinfo?.color;\n const polyDash = s.polyinfo?.dashes;\n const specColor = polyColor && polyColor.length > 0 ? polyColor : color;\n return {\n color: specColor,\n dashPattern: polyDash,\n points: points.map((p) => ({\n lat: p.lat,\n lng: p.lng,\n })),\n };\n });\n\n const computedPolys: ComputedPoly[] = this.polyManager\n .getPolylines()\n .value.map((compPoly) => {\n const polyColor = compPoly.displayInfo.color;\n const polyDash = compPoly.displayInfo.dashes;\n return {\n color: polyColor,\n dashPattern: polyDash,\n points: compPoly.points,\n };\n });\n return pointSetPolys.concat(computedPolys);\n });\n}\n","import { injectable } from 'inversify';\nimport { Map } from 'leaflet';\nimport { MapPoint } from '../../../map-display';\nimport { MapDisplayInfoService } from './map-display-info.interface';\n\n@injectable()\nexport class LeafletMapDisplayInfoService implements MapDisplayInfoService {\n private map?: Map;\n\n setMap(map: Map | undefined): void {\n this.map = map;\n }\n getMapCenter(): MapPoint | undefined {\n if (!this.map) return undefined;\n return this.map.getBounds().getCenter();\n }\n\n setMapCenter(mapPoint: MapPoint): void {\n this.map?.panTo(mapPoint);\n }\n\n getMapZoom(): number | undefined {\n return this.map?.getZoom();\n }\n}\n","\nimport L, { DivIcon, Marker } from 'leaflet';\nimport {\n defineComponent,\n onMounted,\n onUnmounted,\n PropType,\n toRefs,\n watch,\n} from 'vue';\nimport { MapMarkerProps, MapPoint } from '@/map-display';\nimport { Map as LeafletMap } from 'leaflet';\nimport { BASE_PIN_COLOR } from '@/icon-styling';\n\nexport default defineComponent({\n name: 'MapMarker',\n props: {\n point: {\n type: Object as PropType,\n required: true,\n },\n color: {\n type: String,\n },\n iconClass: {\n type: String,\n },\n zIndex: {\n type: Number,\n },\n map: {\n type: Object as PropType,\n required: true,\n },\n },\n render() {\n return [];\n },\n setup(props: MapMarkerProps & { map: LeafletMap }) {\n const { point, color, iconClass, zIndex } = toRefs(props);\n let marker: Marker;\n let icon: DivIcon;\n\n function buildMarker(): Marker {\n const mark = L.marker(L.latLng(point.value.lat, point.value.lng), {\n icon,\n });\n mark.addTo(props.map);\n return mark;\n }\n\n function buildIcon(): DivIcon {\n const styleString =\n color?.value && color.value.length > 0\n ? `style='background-color:${props.color}'`\n : `style='background-color:${BASE_PIN_COLOR}'`;\n const htmlIconClass =\n iconClass?.value && iconClass.value.length > 0\n ? iconClass.value\n : 'bi-circle-fill';\n const IW = 36;\n const IH = Math.sqrt(IW * IW * 2);\n return L.divIcon({\n className: 'custom-div-icon',\n html: ``,\n iconSize: [IW, IH],\n iconAnchor: [IW / 2, IH],\n });\n }\n function reloadMarker() {\n if (marker) {\n marker.setLatLng(L.latLng(point.value.lat, point.value.lng));\n marker.setIcon(icon);\n marker.setZIndexOffset(zIndex?.value ? zIndex.value : 0);\n } else {\n marker = buildMarker();\n }\n }\n function reloadIcon() {\n icon = buildIcon();\n reloadMarker();\n }\n\n onMounted(reloadIcon);\n onUnmounted(() => {\n if (marker) {\n marker.off();\n marker.remove();\n }\n });\n\n watch(point, reloadMarker, { deep: true });\n if (zIndex) watch(zIndex, reloadMarker);\n if (color) watch(color, reloadIcon);\n if (iconClass) watch(iconClass, reloadIcon);\n\n icon = buildIcon();\n marker = buildMarker();\n\n return {\n marker,\n icon,\n };\n },\n});\n","import script from \"./MapMarker.vue?vue&type=script&lang=ts\"\nexport * from \"./MapMarker.vue?vue&type=script&lang=ts\"\n\nimport \"./MapMarker.vue?vue&type=style&index=0&id=46f4eae6&lang=stylus\"\n\nconst __exports__ = script;\n\nexport default __exports__","\nimport { DI } from '@/di';\nimport { DirectionRouterService } from '@/services/map-service/routing/direction-router.interface';\nimport L, { Polyline, Map as LeafletMap } from 'leaflet';\nimport {\n computed,\n ComputedRef,\n defineComponent,\n onUnmounted,\n PropType,\n ref,\n toRefs,\n watch,\n} from 'vue';\nimport { MapPoint, PolyInfoDisplayOptions } from '@/map-display';\n\nexport default defineComponent({\n name: 'MapPolyline',\n props: {\n points: {\n type: Array as PropType,\n required: true,\n },\n options: Object as PropType,\n doRouting: Boolean,\n map: {\n type: Object as PropType,\n required: true,\n },\n },\n render() {\n return [];\n },\n setup(props) {\n const routerService = DI.get(\n DirectionRouterService,\n );\n const { points, options, doRouting } = toRefs(props);\n const routedPoints = ref([]);\n const polyPoints: ComputedRef = computed(() =>\n routedPoints.value.length > 0 ? routedPoints.value : points.value,\n );\n let polyline: Polyline;\n\n function buildPolyline(): Polyline {\n const latlngs = polyPoints.value.map((p) => L.latLng(p.lat, p.lng));\n\n if (polyline) {\n polyline.setLatLngs(latlngs);\n if (options && options.value) polyline.setStyle(options.value);\n } else {\n polyline = new Polyline(latlngs, options?.value);\n polyline.addTo(props.map);\n }\n return polyline;\n }\n\n if (doRouting.value) {\n polyline = new Polyline([], options?.value);\n polyline.addTo(props.map);\n } else {\n polyline = buildPolyline();\n }\n watch([polyPoints, options], buildPolyline, { deep: true });\n\n onUnmounted(() => {\n if (polyline) {\n polyline.off();\n polyline.remove();\n }\n });\n\n async function requestRouting() {\n if (doRouting.value) {\n const newRoutedPoints = await routerService.getRoutedPoints(\n points.value,\n );\n routedPoints.value = newRoutedPoints.flatMap((leg) => leg[0].points);\n }\n }\n\n requestRouting();\n\n watch(\n points,\n () => {\n requestRouting();\n },\n { deep: true },\n );\n\n return { polyline, buildPolyline, routerService };\n },\n});\n","import script from \"./MapPolyline.vue?vue&type=script&lang=ts\"\nexport * from \"./MapPolyline.vue?vue&type=script&lang=ts\"\n\nconst __exports__ = script;\n\nexport default __exports__","\nimport 'leaflet/dist/leaflet.css';\nimport { defineComponent, onMounted, onUnmounted, Ref, shallowRef } from 'vue';\nimport L, { Map } from 'leaflet';\nimport { DI } from '@/di';\nimport { MapTileService } from '@/services/map-service/map-tile.interface';\nimport { PointSetCollapse } from '../../parts/PointSetCollapse';\nimport { PolylineFinder } from '../../parts/PolylineFinder';\nimport { MapDisplayInfoService } from '@/services/map-service/display-info/map-display-info.interface';\nimport { LeafletMapDisplayInfoService } from '@/services/map-service/display-info/leaflet-map-display-info.service';\nimport MapMarker from './MapMarker.vue';\nimport MapPolyline from './MapPolyline.vue';\n\nfunction newMap(tileService: MapTileService): Map {\n let map = L.map('leaflet-map', { zoomControl: false });\n map.setView([51.5073219, -0.1276474], 13);\n L.tileLayer(\n tileService.buildLeafletTileUrl(),\n tileService.buildOptions(),\n ).addTo(map);\n\n let zoomCtrl = new L.Control.Zoom({ position: 'bottomleft' });\n zoomCtrl.addTo(map);\n\n return map;\n}\n\nexport default defineComponent({\n name: 'LeafletMap',\n components: {\n MapMarker,\n MapPolyline,\n },\n setup() {\n const tileService = DI.get(MapTileService);\n const displayInfoService = (function () {\n const raw = DI.get(MapDisplayInfoService);\n if (raw instanceof LeafletMapDisplayInfoService) return raw;\n return undefined;\n })();\n let map: Ref