254 lines
13 KiB
TypeScript
254 lines
13 KiB
TypeScript
import React, { useState } from 'react';
|
|
import { View, Text, Modal, TouchableOpacity, TextInput, ScrollView, Alert } from 'react-native';
|
|
import DateTimePicker, { DateType, useDefaultStyles } from 'react-native-ui-datepicker';
|
|
import { TimeOffRequestType } from '@/types/types';
|
|
import { X } from 'lucide-react-native';
|
|
import { TimePickerModal } from './TimePickerModal';
|
|
import api from '@/utils/api';
|
|
import { formatPickerDate } from '@/utils/dateTime';
|
|
|
|
interface RequestPermitModalProps {
|
|
visible: boolean;
|
|
types: TimeOffRequestType[];
|
|
onClose: () => void;
|
|
onSubmit: (data: any) => void;
|
|
}
|
|
|
|
export default function RequestPermitModal({ visible, types, onClose, onSubmit }: RequestPermitModalProps) {
|
|
const defaultStyles = useDefaultStyles();
|
|
const [type, setType] = useState<TimeOffRequestType>(types[0]); // Default to first type
|
|
const [date, setDate] = useState<string | null>();
|
|
const [range, setRange] = useState<{
|
|
startDate: string | null;
|
|
endDate: string | null;
|
|
}>({ startDate: null, endDate: null });
|
|
|
|
const [showStartPicker, setShowStartPicker] = useState(false);
|
|
const [showEndPicker, setShowEndPicker] = useState(false);
|
|
const [startTime, setStartTime] = useState('');
|
|
const [endTime, setEndTime] = useState('');
|
|
const [message, setMessage] = useState('');
|
|
|
|
// Funzione per resettare le selezioni di data
|
|
const clearCalendar = () => {
|
|
setDate(null);
|
|
setRange({ startDate: null, endDate: null });
|
|
setStartTime(''); setEndTime('');
|
|
setType(types[0]);
|
|
};
|
|
|
|
// Funzione per validare la richiesta
|
|
function validateRequest(type: TimeOffRequestType, date: string | null | undefined, range: { startDate: string | null; endDate: string | null }, startTime: string, endTime: string, message: string): string | null {
|
|
if (!type) return "Seleziona una tipologia di assenza.";
|
|
|
|
if (!message || message.trim() === "") return "Inserisci un messaggio.";
|
|
|
|
if (type.time_required === 0) {
|
|
if (!range.startDate) return "Seleziona una data di inizio.";
|
|
return null;
|
|
}
|
|
|
|
if (!date) return "Seleziona una data.";
|
|
if (!startTime || !endTime) return "Seleziona gli orari.";
|
|
if (startTime >= endTime) return "L'orario di fine deve essere successivo a quello di inizio.";
|
|
|
|
return null;
|
|
}
|
|
|
|
// Funzione per inviare la richiesta alla API
|
|
const saveRequest = async (requestData: any) => {
|
|
try {
|
|
const response = await api.post('/time-off-request/save-request', requestData);
|
|
|
|
if (response.data.status === 'success') {
|
|
Alert.alert('Successo', response.data.message || 'La tua richiesta è stata inviata con successo.');
|
|
} else {
|
|
Alert.alert('Errore', response.data.message || 'Impossibile inviare la richiesta.');
|
|
}
|
|
} catch (error: any) {
|
|
console.error('Errore nell\'invio della richiesta:', error);
|
|
throw new Error('Impossibile inviare la richiesta.');
|
|
}
|
|
};
|
|
|
|
// Funzione per inviare la richiesta
|
|
const handleSubmit = async () => {
|
|
const error = validateRequest(type, date, range, startTime, endTime, message);
|
|
if (error) return Alert.alert("Errore", error);
|
|
|
|
const requestData = {
|
|
id_type: type.id,
|
|
start_date: type.time_required === 0 ? range.startDate : date,
|
|
end_date: type.time_required === 0 ? range.endDate : null,
|
|
start_time: type.time_required === 1 ? startTime : null,
|
|
end_time: type.time_required === 1 ? endTime : null,
|
|
message: message || ""
|
|
};
|
|
|
|
try {
|
|
await saveRequest(requestData);
|
|
onSubmit(requestData); // TODO: Gestire risposta e controllare fetch in index?
|
|
onClose();
|
|
} catch (e) {
|
|
Alert.alert("Errore", "Impossibile inviare la richiesta.");
|
|
}
|
|
};
|
|
|
|
return (
|
|
<Modal
|
|
visible={visible}
|
|
transparent={true}
|
|
animationType="slide"
|
|
statusBarTranslucent
|
|
>
|
|
<View className="flex-1 bg-black/60 justify-end sm:justify-center">
|
|
<View className="bg-white w-full rounded-t-[2.5rem] p-6 shadow-2xl h-[85%] sm:h-auto">
|
|
{/* Header Modale */}
|
|
<View className="flex-row justify-between items-center mb-6">
|
|
<Text className="text-2xl font-bold text-gray-800">Nuova Richiesta</Text>
|
|
<TouchableOpacity onPress={onClose} className="p-2 bg-gray-100 rounded-full">
|
|
<X size={24} color="#4b5563" />
|
|
</TouchableOpacity>
|
|
</View>
|
|
|
|
<ScrollView showsVerticalScrollIndicator={false} contentContainerStyle={{ paddingBottom: 40 }}>
|
|
<View className="space-y-6">
|
|
{/* Tipologia */}
|
|
<View className='mb-6'>
|
|
<Text className="text-lg font-bold text-gray-700 mb-3">Tipologia Assenza</Text>
|
|
<ScrollView
|
|
horizontal
|
|
showsHorizontalScrollIndicator={false}
|
|
contentContainerStyle={{ paddingHorizontal: 0, gap: 12 }}
|
|
>
|
|
{types.map((t) => (
|
|
<TouchableOpacity
|
|
key={t.id}
|
|
onPress={() => setType(t)}
|
|
className={`py-4 px-5 rounded-xl border-2 items-center justify-center ${type?.id === t.id ? 'border-[#099499] bg-teal-50' : 'border-gray-100 bg-white'
|
|
}`}
|
|
>
|
|
<Text className={`text-sm font-bold ${type?.id === t.id ? 'text-[#099499]' : 'text-gray-500'}`}>
|
|
{t.name}
|
|
</Text>
|
|
</TouchableOpacity>
|
|
))}
|
|
</ScrollView>
|
|
</View>
|
|
|
|
{/* Date and Time Selection */}
|
|
{type?.time_required === 0 ? (
|
|
<DateTimePicker
|
|
mode="range"
|
|
startDate={range.startDate}
|
|
endDate={range.endDate}
|
|
onChange={(params) => {
|
|
setRange({
|
|
startDate: params.startDate ? formatPickerDate(params.startDate) : null,
|
|
endDate: params.endDate ? formatPickerDate(params.endDate) : null
|
|
})
|
|
}}
|
|
locale='it'
|
|
styles={{
|
|
...defaultStyles,
|
|
selected: { backgroundColor: '#099499' }
|
|
}}
|
|
/>
|
|
) : (
|
|
<DateTimePicker
|
|
mode="single"
|
|
date={date}
|
|
onChange={({ date }) => setDate(date ? formatPickerDate(date) : null)}
|
|
locale='it'
|
|
styles={{
|
|
...defaultStyles,
|
|
selected: { backgroundColor: '#099499' }
|
|
}}
|
|
/>
|
|
)}
|
|
|
|
<View className='flex-column bg-orange-50 rounded-xl border border-orange-100 mb-6'>
|
|
{type?.time_required === 1 && (
|
|
<View>
|
|
<View className="flex-row gap-4 p-4">
|
|
<View className="flex-1">
|
|
<Text className="text-sm font-bold text-orange-800 mb-2 uppercase">Dalle Ore</Text>
|
|
<TouchableOpacity onPress={() => setShowStartPicker(true)}>
|
|
<TextInput
|
|
placeholder="09:00"
|
|
className="w-full p-3 bg-white rounded-lg border border-orange-200 font-bold text-gray-800 text-center"
|
|
value={startTime}
|
|
onChangeText={setStartTime}
|
|
editable={false}
|
|
/>
|
|
</TouchableOpacity>
|
|
</View>
|
|
<View className="flex-1">
|
|
<Text className="text-sm font-bold text-orange-800 mb-2 uppercase">Alle Ore</Text>
|
|
<TouchableOpacity onPress={() => setShowEndPicker(true)}>
|
|
<TextInput
|
|
placeholder="18:00"
|
|
className="w-full p-3 bg-white rounded-lg border border-orange-200 font-bold text-gray-800 text-center"
|
|
value={endTime}
|
|
onChangeText={setEndTime}
|
|
editable={false}
|
|
/>
|
|
</TouchableOpacity>
|
|
</View>
|
|
</View>
|
|
|
|
<TimePickerModal
|
|
visible={showStartPicker}
|
|
initialDate={new Date()}
|
|
title="Seleziona Ora Inizio"
|
|
onConfirm={(time) => setStartTime(time)}
|
|
onClose={() => setShowStartPicker(false)}
|
|
/>
|
|
<TimePickerModal
|
|
visible={showEndPicker}
|
|
initialDate={new Date()}
|
|
title="Seleziona Ora Fine"
|
|
onConfirm={(time) => setEndTime(time)}
|
|
onClose={() => setShowEndPicker(false)}
|
|
/>
|
|
</View>
|
|
)}
|
|
{/* TODO: Trasformare message in una select? - Predefinito per alcuni tipi */}
|
|
<View className="p-4">
|
|
<Text className="text-sm font-bold text-orange-800 mb-2 uppercase">Motivo</Text>
|
|
<TextInput
|
|
placeholder="Scrivi qui il motivo..."
|
|
className="w-full px-3 py-4 bg-white font-bold text-gray-800 rounded-lg border border-orange-200 text-gray-800"
|
|
textAlignVertical="top"
|
|
value={message}
|
|
onChangeText={setMessage}
|
|
/>
|
|
</View>
|
|
</View>
|
|
|
|
{/* Azioni */}
|
|
<View className="flex-row gap-4">
|
|
<TouchableOpacity
|
|
onPress={() => {
|
|
clearCalendar();
|
|
onClose();
|
|
}}
|
|
className="flex-1 py-4 bg-gray-200 rounded-2xl shadow-sm active:bg-gray-300"
|
|
>
|
|
<Text className="text-gray-700 text-center font-bold text-lg">Annulla Richiesta</Text>
|
|
</TouchableOpacity>
|
|
|
|
<TouchableOpacity
|
|
onPress={handleSubmit}
|
|
className="flex-1 py-4 bg-[#099499] rounded-2xl shadow-lg active:scale-[0.98]"
|
|
>
|
|
<Text className="text-white text-center font-bold text-lg">Invia Richiesta</Text>
|
|
</TouchableOpacity>
|
|
</View>
|
|
</View>
|
|
</ScrollView>
|
|
</View>
|
|
</View>
|
|
</Modal>
|
|
);
|
|
}; |