Files
SIB/frontend/src/components/BusStopEditor.vue

268 lines
6.5 KiB
Vue

<template>
<div class="editor-container">
<div class="editor-content">
<h2>{{ isEditing ? 'Editar Parada' : 'Crear Parada' }}</h2>
<div class="form-grid">
<div class="form-group">
<label>Nombre</label>
<input v-model="formData.name" type="text" placeholder="Nombre de la Parada" />
</div>
<div class="form-group">
<label>Tipo</label>
<select v-model="formData.stop_type">
<option value="REGULAR">Regular</option>
<option value="TERMINAL">Terminal</option>
<option value="EXPRESS_ONLY">Solo Expreso</option>
</select>
</div>
<div class="checkbox-group">
<label>
<input type="checkbox" v-model="formData.has_shelter"> Techo
</label>
<label>
<input type="checkbox" v-model="formData.has_seating"> Asientos
</label>
<label>
<input type="checkbox" v-model="formData.is_accessible"> Accesible
</label>
</div>
<div class="gps-section">
<div class="coordinates">
Lat: {{ formData.latitude.toFixed(6) }}, Lon: {{ formData.longitude.toFixed(6) }}
</div>
<button @click="getCurrentLocation" class="gps-button" :disabled="isLoadingGps">
<span class="material-icons">my_location</span>
{{ isLoadingGps ? 'Localizando...' : 'Usar GPS Preciso' }}
</button>
</div>
</div>
<div class="map-wrapper">
<div id="editor-map" class="editor-map"></div>
<div v-if="!isMapLoaded" class="map-loading">Cargando Mapa...</div>
</div>
<div class="actions">
<button @click="$emit('cancel')" class="cancel-button">Cancelar</button>
<button @click="handleSave" class="save-button" :disabled="isSaving">
{{ isSaving ? 'Guardando...' : 'Guardar Parada' }}
</button>
</div>
<div v-if="error" class="error-message">{{ error }}</div>
</div>
</div>
</template>
<script setup lang="ts">
import { ref, onMounted, watch } from 'vue'
import { Geolocation } from '@capacitor/geolocation'
import { useGoogleMaps } from '@/composables/useGoogleMaps'
import type { BusStop } from '@/types'
const props = defineProps<{
initialStop?: BusStop | null
}>()
const emit = defineEmits(['save', 'cancel'])
const isEditing = ref(!!props.initialStop)
const isSaving = ref(false)
const isLoadingGps = ref(false)
const error = ref<string | null>(null)
const formData = ref({
name: '',
latitude: 8.4284, // Default (David)
longitude: -82.4309,
stop_type: 'REGULAR' as import('@/types').StopType,
has_shelter: false,
has_seating: false,
is_accessible: false,
city: 'David', // Default
})
const { initMap, addMarker, setCenter, isLoaded: isMapLoaded } = useGoogleMaps()
let marker: google.maps.Marker | null = null
onMounted(async () => {
if (props.initialStop) {
formData.value = {
name: props.initialStop.name,
latitude: props.initialStop.latitude,
longitude: props.initialStop.longitude,
stop_type: props.initialStop.stop_type,
has_shelter: props.initialStop.has_shelter,
has_seating: props.initialStop.has_seating,
is_accessible: props.initialStop.is_accessible,
city: props.initialStop.city || 'David',
}
}
// Initialize map
initMap('editor-map', { lat: formData.value.latitude, lng: formData.value.longitude }, 16)
updateMarker()
})
// Watch for map load to add marker if missed
watch(isMapLoaded, (loaded) => {
if (loaded) {
updateMarker()
}
})
function updateMarker() {
if (!isMapLoaded.value) return
if (marker) {
marker.setPosition({ lat: formData.value.latitude, lng: formData.value.longitude })
} else {
marker = addMarker(
{ lat: formData.value.latitude, lng: formData.value.longitude },
{
draggable: true,
onDragEnd: (pos) => {
formData.value.latitude = pos.lat
formData.value.longitude = pos.lng
}
}
)
}
setCenter(formData.value.latitude, formData.value.longitude)
}
async function getCurrentLocation() {
isLoadingGps.value = true
try {
const coordinates = await Geolocation.getCurrentPosition({
enableHighAccuracy: true,
timeout: 10000
})
formData.value.latitude = coordinates.coords.latitude
formData.value.longitude = coordinates.coords.longitude
updateMarker()
} catch (e) {
console.error('GPS Error', e)
error.value = 'Error al obtener la ubicación GPS. Asegúrate de dar los permisos.'
} finally {
isLoadingGps.value = false
}
}
function handleSave() {
if (!formData.value.name) {
error.value = 'El nombre es obligatorio'
return
}
emit('save', {
...formData.value,
id: props.initialStop?.id
})
}
</script>
<style scoped>
.editor-container {
background: white;
padding: 20px;
border-radius: 8px;
max-width: 800px;
margin: 0 auto;
box-shadow: 0 4px 6px rgba(0,0,0,0.1);
}
.form-grid {
display: grid;
gap: 16px;
margin-bottom: 20px;
}
.form-group {
display: flex;
flex-direction: column;
gap: 4px;
}
input, select {
padding: 8px;
border: 1px solid #ddd;
border-radius: 4px;
}
.checkbox-group {
display: flex;
gap: 16px;
}
.gps-section {
display: flex;
justify-content: space-between;
align-items: center;
background: #f8f9fa;
padding: 10px;
border-radius: 4px;
}
.gps-button {
display: flex;
align-items: center;
gap: 6px;
background: #2ecc71;
color: white;
border: none;
padding: 8px 12px;
border-radius: 4px;
cursor: pointer;
}
.gps-button:disabled {
opacity: 0.6;
}
.map-wrapper {
height: 300px;
background: #eee;
margin-bottom: 20px;
position: relative;
}
.editor-map {
width: 100%;
height: 100%;
}
.actions {
display: flex;
justify-content: flex-end;
gap: 10px;
}
.save-button {
background: #3498db;
color: white;
border: none;
padding: 10px 20px;
border-radius: 4px;
cursor: pointer;
}
.cancel-button {
background: #95a5a6;
color: white;
border: none;
padding: 10px 20px;
border-radius: 4px;
cursor: pointer;
}
.error-message {
color: red;
margin-top: 10px;
}
</style>