MapLeaflet.client.vue 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. <template>
  2. <div class="map-container">
  3. <v-skeleton-loader type="image" v-if="pending"></v-skeleton-loader>
  4. <LMap
  5. v-show="!pending"
  6. style="height: 350px"
  7. :zoom="zoom"
  8. :center="position"
  9. :use-global-leaflet="false"
  10. >
  11. <LTileLayer
  12. url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
  13. attribution="&amp;copy; <a href=&quot;https://www.openstreetmap.org/&quot;>OpenStreetMap</a> contributors"
  14. layer-type="base"
  15. name="OpenStreetMap"
  16. />
  17. <LMarker
  18. :lat-lng="position"
  19. @update:latLng="onPositionUpdate($event)"
  20. draggable />
  21. </LMap>
  22. <v-btn
  23. prepend-icon="fas fa-location-dot"
  24. class="mt-3"
  25. v-if="searchButton"
  26. @click="search()"
  27. >
  28. {{$t('search_gps_button')}}
  29. </v-btn>
  30. <div v-if="!pending && gpsResponses.length > 0">
  31. <div v-for="(gpsResponse, key) in gpsResponses" class="address_choices" @click="addressChoice(key)">
  32. {{gpsResponse['displayName']}}
  33. <v-btn
  34. prepend-icon="fas fa-map-location"
  35. @click="addressChoice(key)"
  36. >Choisir</v-btn>
  37. </div>
  38. </div>
  39. </div>
  40. </template>
  41. <script setup lang="ts">
  42. import 'leaflet/dist/leaflet.css'
  43. import { LMap, LTileLayer, LMarker } from '@vue-leaflet/vue-leaflet'
  44. import {type ComputedRef, defineProps, type PropType} from 'vue'
  45. import {LatLng, type PointTuple} from 'leaflet'
  46. import {useAp2iRequestService} from "~/composables/data/useAp2iRequestService";
  47. import UrlUtils from "~/services/utils/urlUtils";
  48. import type {AnyJson, CollectionResponsePromise} from "~/types/data";
  49. import Country from "~/models/Core/Country";
  50. import {useEntityManager} from "~/composables/data/useEntityManager";
  51. const props = defineProps({
  52. latitude: {
  53. type: Number as PropType<number | null>,
  54. required: true,
  55. },
  56. longitude: {
  57. type: Number as PropType<number | null>,
  58. required: true,
  59. },
  60. streetAddress:{
  61. type: String as PropType<string | null>,
  62. required: false
  63. },
  64. streetAddressSecond:{
  65. type: String as PropType<string | null>,
  66. required: false
  67. },
  68. streetAddressThird:{
  69. type: String as PropType<string | null>,
  70. required: false
  71. },
  72. postalCode:{
  73. type: String as PropType<string | null>,
  74. required: false
  75. },
  76. addressCity:{
  77. type: String as PropType<string | null>,
  78. required: false
  79. },
  80. addressCountryId:{
  81. type: Number as PropType<number | null>,
  82. required: false
  83. },
  84. searchButton:{
  85. type: Boolean,
  86. required: false,
  87. default: false
  88. }
  89. })
  90. const {apiRequestService, pending} = useAp2iRequestService()
  91. const { em } = useEntityManager()
  92. const position:ComputedRef<PointTuple> = computed(()=>{
  93. return [props.latitude || 12, props.longitude || 12]
  94. })
  95. const zoom = ref(12)
  96. const emit = defineEmits(['update:latitude', 'update:longitude'])
  97. const onPositionUpdate = (event: LatLng):void => {
  98. emit('update:latitude', event.lat)
  99. emit('update:longitude', event.lng)
  100. }
  101. const gpsResponses:Ref<Array<AnyJson>> = ref([])
  102. const search = async () => {
  103. gpsResponses.value = []
  104. const baseUrl = UrlUtils.join('api', 'gps-coordinate-searching')
  105. const query:AnyJson = {
  106. 'streetAddress': props.streetAddress,
  107. 'streetAddressSecond': props.streetAddressSecond,
  108. 'streetAddressThird': props.streetAddressThird,
  109. 'cp': props.postalCode,
  110. 'city': props.addressCity
  111. }
  112. if(props.addressCountryId){
  113. const country:Country = em.find(Country, props.addressCountryId)
  114. query['country'] = country?.name
  115. }
  116. const url = UrlUtils.addQuery(baseUrl, query)
  117. const responses:CollectionResponsePromise = await apiRequestService.get(url)
  118. if(responses['member'].length > 0){
  119. onPositionUpdate(new LatLng(responses['member'][0]['latitude'], responses['member'][0]['longitude']))
  120. if(responses['member'].length > 1){
  121. zoom.value = 6
  122. gpsResponses.value = responses['member']
  123. }else{
  124. zoom.value = 12
  125. }
  126. }
  127. }
  128. const addressChoice = (key:number):void => {
  129. zoom.value = 12
  130. onPositionUpdate(new LatLng(gpsResponses.value[key]['latitude'] as number, gpsResponses.value[key]['longitude'] as number))
  131. }
  132. </script>
  133. <style scoped lang="scss">
  134. .address_choices {
  135. cursor: pointer;
  136. width: 60%;
  137. display: flex;
  138. justify-content: space-between;
  139. align-items: center;
  140. padding: 0.75rem 1rem;
  141. border-radius: 0.5rem;
  142. margin-top: 0.5rem;
  143. background-color: #f9f9f9;
  144. transition: background-color 0.2s ease;
  145. &:hover {
  146. background-color: #eef3ff;
  147. }
  148. .v-btn {
  149. flex-shrink: 0;
  150. }
  151. }
  152. :deep(.v-skeleton-loader__image) {
  153. height: 350px;
  154. }
  155. :deep(.map_wrap) {
  156. height: 350px;
  157. }
  158. </style>