MapLeaflet.client.vue 5.1 KB

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