Added Charts
This commit is contained in:
@@ -0,0 +1,381 @@
|
||||
<script setup lang="ts">
|
||||
import MainLayout from '@/Layouts/MainLayout.vue'
|
||||
import GlassBox from '@/Components/FlightsGoneBy/GlassBox.vue'
|
||||
import { Head, useForm } from '@inertiajs/vue3'
|
||||
import AirlineSearchBox from '@/Components/FlightsGoneBy/AirlineSearchBox.vue'
|
||||
import AircraftSearchBox from '@/Components/FlightsGoneBy/AircraftSearchBox.vue'
|
||||
import AirportSearchBox from '@/Components/FlightsGoneBy/AirportSearchBox.vue'
|
||||
import { ref } from 'vue'
|
||||
|
||||
defineOptions({ layout: MainLayout })
|
||||
|
||||
const props = defineProps<{
|
||||
flight?: {
|
||||
id: number
|
||||
flight_number: string
|
||||
departure_date: string
|
||||
arrival_date: string
|
||||
aircraft_registration: string
|
||||
seat_number: string
|
||||
note: string
|
||||
auto_update: boolean
|
||||
seat_type: { value: number; title: string } | null
|
||||
flight_class: { value: number; title: string } | null
|
||||
flight_reason: { value: number; title: string } | null
|
||||
airline_options: { value: number; title: string }[]
|
||||
from_options: { value: number; title: string; country_code: string }[]
|
||||
to_options: { value: number; title: string; country_code: string }[]
|
||||
aircraft_options: { value: number; title: string }[]
|
||||
}
|
||||
seat_types: { value: number; title: string }[]
|
||||
flight_classes: { value: number; title: string }[]
|
||||
flight_reasons: { value: number; title: string }[]
|
||||
}>()
|
||||
|
||||
const isEdit = !!props.flight
|
||||
|
||||
// ── Flight number lookup ──────────────────────────────────────────────────────
|
||||
|
||||
const flightNumber = ref(props.flight?.flight_number ?? '')
|
||||
const lookupLoading = ref(false)
|
||||
const lookupError = ref<string | null>(null)
|
||||
const lookupComplete = ref(isEdit)
|
||||
|
||||
interface LookupResult {
|
||||
airline_options: { value: number; title: string }[]
|
||||
from_options: { value: number; title: string; country_code: string }[]
|
||||
to_options: { value: number; title: string; country_code: string }[]
|
||||
aircraft_options: { value: number; title: string }[]
|
||||
}
|
||||
|
||||
const lookupResult = ref<LookupResult | null>(null)
|
||||
const lookupKey = ref(0)
|
||||
async function lookupFlight() {
|
||||
if (!flightNumber.value.trim()) return
|
||||
lookupLoading.value = true
|
||||
lookupError.value = null
|
||||
lookupResult.value = null
|
||||
|
||||
try {
|
||||
const response = await fetch(`${route('flights.lookup')}?number=${encodeURIComponent(flightNumber.value.trim())}`, {
|
||||
headers: { Accept: 'application/json' },
|
||||
})
|
||||
const data = await response.json()
|
||||
if (!response.ok) {
|
||||
lookupError.value = data.message ?? 'Lookup failed.'
|
||||
return
|
||||
}
|
||||
lookupResult.value = data
|
||||
lookupComplete.value = true
|
||||
|
||||
if (data.airline_options?.length) {
|
||||
airlineOptionsData.value = data.airline_options
|
||||
if (!form.airline) form.airline = data.airline_options[0]
|
||||
}
|
||||
if (data.from_options?.length) {
|
||||
fromOptionsData.value = data.from_options
|
||||
if (data.from_options.length === 1 && !form.from) form.from = data.from_options[0]
|
||||
}
|
||||
if (data.to_options?.length) {
|
||||
toOptionsData.value = data.to_options
|
||||
if (data.to_options.length === 1 && !form.to) form.to = data.to_options[0]
|
||||
}
|
||||
if (data.aircraft_options?.length) {
|
||||
aircraftOptionsData.value = data.aircraft_options
|
||||
if (data.aircraft_options.length === 1 && !form.aircraft) form.aircraft = data.aircraft_options[0]
|
||||
}
|
||||
|
||||
lookupKey.value++
|
||||
} catch (e) {
|
||||
lookupError.value = String(e)
|
||||
} finally {
|
||||
lookupLoading.value = false
|
||||
}
|
||||
}
|
||||
|
||||
// ── Form ──────────────────────────────────────────────────────────────────────
|
||||
|
||||
const form = useForm({
|
||||
flight_number: props.flight?.flight_number ?? '',
|
||||
departure_date: props.flight?.departure_date ?? '',
|
||||
arrival_date: props.flight?.arrival_date ?? '',
|
||||
from: props.flight?.from_options[0] ?? null as { value: number; title: string; country_code: string } | null,
|
||||
to: props.flight?.to_options[0] ?? null as { value: number; title: string; country_code: string } | null,
|
||||
airline: props.flight?.airline_options[0] ?? null as { value: number; title: string } | null,
|
||||
aircraft: props.flight?.aircraft_options[0] ?? null as { value: number; title: string } | null,
|
||||
aircraft_registration: props.flight?.aircraft_registration ?? '',
|
||||
seat_number: props.flight?.seat_number ?? '',
|
||||
seat_type: props.flight?.seat_type ?? props.seat_types[0],
|
||||
flight_class: props.flight?.flight_class ?? props.flight_classes[0],
|
||||
flight_reason: props.flight?.flight_reason ?? props.flight_reasons[0],
|
||||
note: props.flight?.note ?? '',
|
||||
auto_update: props.flight?.auto_update ?? false,
|
||||
})
|
||||
|
||||
const submitForm = useForm({
|
||||
flight_number: '' as string | null,
|
||||
departure_date: '' as string | null,
|
||||
arrival_date: '' as string | null,
|
||||
from_id: null as number | null,
|
||||
to_id: null as number | null,
|
||||
airline_id: null as number | null,
|
||||
aircraft_id: null as number | null,
|
||||
aircraft_registration: '' as string | null,
|
||||
seat_number: '' as string | null,
|
||||
seat_type_id: null as number | null,
|
||||
flight_class_id: null as number | null,
|
||||
flight_reason_id: null as number | null,
|
||||
note: '' as string | null,
|
||||
auto_update: false,
|
||||
})
|
||||
|
||||
function submit() {
|
||||
submitForm.flight_number = form.flight_number
|
||||
submitForm.departure_date = form.departure_date
|
||||
submitForm.arrival_date = form.arrival_date
|
||||
submitForm.from_id = form.from?.value ?? null
|
||||
submitForm.to_id = form.to?.value ?? null
|
||||
submitForm.airline_id = form.airline?.value ?? null
|
||||
submitForm.aircraft_id = form.aircraft?.value ?? null
|
||||
submitForm.aircraft_registration = form.aircraft_registration
|
||||
submitForm.seat_number = form.seat_number
|
||||
submitForm.seat_type_id = form.seat_type?.value ?? null
|
||||
submitForm.flight_class_id = form.flight_class?.value ?? null
|
||||
submitForm.flight_reason_id = form.flight_reason?.value ?? null
|
||||
submitForm.note = form.note
|
||||
submitForm.auto_update = form.auto_update
|
||||
|
||||
if (isEdit) {
|
||||
submitForm.put(route('flights.update', { flight: props.flight!.id }))
|
||||
} else {
|
||||
submitForm.post(route('flights.store'))
|
||||
}
|
||||
}
|
||||
|
||||
// ── Prefilled options ─────────────────────────────────────────────────────────
|
||||
|
||||
const airlineOptionsData = ref<{ value: number; title: string }[]>(props.flight?.airline_options ?? [])
|
||||
const fromOptionsData = ref<{ value: number; title: string; country_code: string }[]>(props.flight?.from_options ?? [])
|
||||
const toOptionsData = ref<{ value: number; title: string; country_code: string }[]>(props.flight?.to_options ?? [])
|
||||
const aircraftOptionsData = ref<{ value: number; title: string }[]>(props.flight?.aircraft_options ?? [])
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<Head :title="isEdit ? 'Edit Flight' : 'Add Flight'" />
|
||||
|
||||
<GlassBox
|
||||
:title="isEdit ? 'Edit Flight' : 'Add Flight'"
|
||||
:blurb="isEdit
|
||||
? 'Update the details for this flight.'
|
||||
: 'Enter a flight number then press Look Up to continue.'"
|
||||
>
|
||||
<v-form style="width: 100%">
|
||||
<v-container>
|
||||
|
||||
<!-- ── Flight number + lookup ────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12">
|
||||
<div class="d-flex ga-3 align-start">
|
||||
<v-text-field
|
||||
v-model="flightNumber"
|
||||
label="Flight Number"
|
||||
placeholder="e.g. QF1"
|
||||
hide-details
|
||||
@keydown.enter="lookupFlight"
|
||||
/>
|
||||
<v-btn
|
||||
:loading="lookupLoading"
|
||||
:disabled="!flightNumber.trim()"
|
||||
size="large"
|
||||
style="height: 56px"
|
||||
@click="lookupFlight"
|
||||
>
|
||||
Look Up
|
||||
</v-btn>
|
||||
</div>
|
||||
<div v-if="lookupError" class="text-error text-caption mt-1">
|
||||
{{ lookupError }}
|
||||
</div>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── Departure + Arrival datetime ──────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12" md="6">
|
||||
<v-text-field
|
||||
v-model="form.departure_date"
|
||||
label="Departure Date & Time"
|
||||
type="datetime-local"
|
||||
:disabled="!lookupComplete"
|
||||
:error-messages="submitForm.errors.departure_date"
|
||||
/>
|
||||
</v-col>
|
||||
<v-col cols="12" md="6">
|
||||
<v-text-field
|
||||
v-model="form.arrival_date"
|
||||
label="Arrival Date & Time"
|
||||
type="datetime-local"
|
||||
:disabled="!lookupComplete"
|
||||
:error-messages="submitForm.errors.arrival_date"
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── From ──────────────────────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12">
|
||||
<AirportSearchBox
|
||||
v-model="form.from"
|
||||
label="From"
|
||||
:prefilled-options="fromOptionsData"
|
||||
:error-messages="submitForm.errors.from_id"
|
||||
:disabled="!lookupComplete"
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── To ────────────────────────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12">
|
||||
<AirportSearchBox
|
||||
v-model="form.to"
|
||||
label="To"
|
||||
:prefilled-options="toOptionsData"
|
||||
:error-messages="submitForm.errors.to_id"
|
||||
:disabled="!lookupComplete"
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── Airline ────────────────────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12">
|
||||
<AirlineSearchBox
|
||||
:key="`airline-${lookupKey}`"
|
||||
v-model="form.airline"
|
||||
:prefilled-options="airlineOptionsData"
|
||||
:error-messages="submitForm.errors.airline_id"
|
||||
:disabled="!lookupComplete"
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── Aircraft ───────────────────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12">
|
||||
<AircraftSearchBox
|
||||
v-model="form.aircraft"
|
||||
:prefilled-options="aircraftOptionsData"
|
||||
:error-messages="submitForm.errors.aircraft_id"
|
||||
:disabled="!lookupComplete"
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── Registration + Flight class ────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12" md="6">
|
||||
<v-text-field
|
||||
v-model="form.aircraft_registration"
|
||||
label="Aircraft Registration"
|
||||
placeholder="e.g. VH-OQA"
|
||||
:disabled="!lookupComplete"
|
||||
:error-messages="submitForm.errors.aircraft_registration"
|
||||
/>
|
||||
</v-col>
|
||||
<v-col cols="12" md="6">
|
||||
<v-select
|
||||
v-model="form.flight_class"
|
||||
label="Flight Class"
|
||||
:items="flight_classes"
|
||||
:disabled="!lookupComplete"
|
||||
:error-messages="submitForm.errors.flight_class_id"
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── Seat number + Seat type ────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12" md="6">
|
||||
<v-text-field
|
||||
v-model="form.seat_number"
|
||||
label="Seat Number"
|
||||
placeholder="e.g. 12A"
|
||||
:disabled="!lookupComplete"
|
||||
:error-messages="submitForm.errors.seat_number"
|
||||
/>
|
||||
</v-col>
|
||||
<v-col cols="12" md="6">
|
||||
<v-select
|
||||
v-model="form.seat_type"
|
||||
label="Seat Type"
|
||||
:items="seat_types"
|
||||
:disabled="!lookupComplete"
|
||||
:error-messages="submitForm.errors.seat_type_id"
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── Flight reason ──────────────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12" md="6">
|
||||
<v-select
|
||||
v-model="form.flight_reason"
|
||||
label="Flight Reason"
|
||||
:items="flight_reasons"
|
||||
:disabled="!lookupComplete"
|
||||
:error-messages="submitForm.errors.flight_reason_id"
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── Note ──────────────────────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12">
|
||||
<v-textarea
|
||||
v-model="form.note"
|
||||
label="Note"
|
||||
placeholder="Any additional notes…"
|
||||
:disabled="!lookupComplete"
|
||||
:error-messages="submitForm.errors.note"
|
||||
rows="3"
|
||||
auto-grow
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── Auto update ────────────────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12">
|
||||
<v-checkbox
|
||||
v-model="form.auto_update"
|
||||
label="Automatically update aircraft details when flight departs."
|
||||
:disabled="!lookupComplete"
|
||||
hide-details
|
||||
density="compact"
|
||||
/>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<!-- ── Submit ─────────────────────────────────────────────── -->
|
||||
<v-row>
|
||||
<v-col cols="12">
|
||||
<v-btn
|
||||
block
|
||||
size="large"
|
||||
:loading="submitForm.processing"
|
||||
:disabled="!lookupComplete || submitForm.processing"
|
||||
@click="submit"
|
||||
>
|
||||
{{ isEdit ? 'Save Changes' : 'Add Flight' }}
|
||||
</v-btn>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
</v-container>
|
||||
</v-form>
|
||||
</GlassBox>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
</style>
|
||||
Reference in New Issue
Block a user