schema([ Section::make('') ->schema([ Forms\Components\Select::make('plant_id') ->relationship('plant', 'name') ->required() // ->nullable() ->reactive() ->disabled(fn (Get $get) => !empty($get('id'))) // ->afterStateUpdated(fn ($set) => $set('block_id', null) & $set('name', null) & $set('start_time', null) & $set('duration', null) & $set('end_time', null)) ->afterStateUpdated(function ($state, callable $set, callable $get) { $plantId = $get('plant_id'); $set('block_id', null); if (!$plantId) { $set('sPlantError', 'Please select a plant first.'); return; } else { $set('sPlantError', null); } }) ->extraAttributes(fn ($get) => [ 'class' => $get('sPlantError') ? 'border-red-500' : '', ]) ->hint(fn ($get) => $get('sPlantError') ? $get('sPlantError') : null) ->hintColor('danger'), Forms\Components\Select::make('block_id') ->relationship('block', 'name') ->required() // ->nullable() ->reactive() ->disabled(fn (Get $get) => !empty($get('id'))) // ->options(fn (callable $get) => // \App\Models\Block::where('plant_id', $get('plant_id')) // ->pluck('name', 'id') // ->toArray() // Convert collection to array // ) ->options(function (callable $get) { if (!$get('plant_id')) { return []; } return \App\Models\Block::where('plant_id', $get('plant_id')) ->pluck('name', 'id') ->toArray(); }) // ->afterStateUpdated(fn ($set) => $set('name', null)) ->afterStateUpdated(function ($state, callable $set, callable $get) { $blockId = $get('block_id'); $set('name', null); if (!$blockId) { $set('sBlockError', 'Please select a block first.'); return; } else { $set('sBlockError', null); } }) ->extraAttributes(fn ($get) => [ 'class' => $get('sBlockError') ? 'border-red-500' : '', ]) ->hint(fn ($get) => $get('sBlockError') ? $get('sBlockError') : null) ->hintColor('danger'), Forms\Components\TextInput::make('name') ->placeholder('Scan the valid name') ->required() ->reactive() ->afterStateUpdated(function ($state, callable $set, callable $get) { $nameId = $get('name'); $set('start_time', null); if (!$nameId) { $set('sNameError', 'Scan the valid name.'); return; } else { $set('sNameError', null); } }) ->extraAttributes(fn ($get) => [ 'class' => $get('sNameError') ? 'border-red-500' : '', ]) ->hint(fn ($get) => $get('sNameError') ? $get('sNameError') : null) ->hintColor('danger'), Forms\Components\TimePicker::make('start_time') ->required() ->label('Start Time') ->live() // ->afterStateUpdated(fn (callable $set, callable $get, $state) => // $set('end_time', self::calculateEndTime($state, $get('duration'))) // ) ->reactive() ->afterStateUpdated(function ($state, callable $set, callable $get) { $nameId = $get('start_time'); // $set('duration', null); $set('end_time', self::calculateEndTime($state, $get('duration'))); if (!$nameId) { $set('sStartTimeError', 'Choose the valid start time.'); return; } else { $set('sStartTimeError', null); } }) ->extraAttributes(fn ($get) => [ 'class' => $get('sStartTimeError') ? 'border-red-500' : '', ]) ->hint(fn ($get) => $get('sStartTimeError') ? $get('sStartTimeError') : null) ->hintColor('danger'), Forms\Components\TextInput::make('duration') ->required() ->placeholder('Scan the valid duration') ->numeric() ->inputMode('decimal') ->minValue(0.01) // Minimum valid duration ->lazy() // ->afterStateUpdated(fn (callable $set, callable $get, $state) => // $set('end_time', self::calculateEndTime($get('start_time'), $state)) // ) ->reactive() ->afterStateUpdated(function ($state, callable $set, callable $get) { $duration = $get('duration'); // $set('end_time', null); $set('end_time', self::calculateEndTime($get('start_time'), $state)); if (!$duration) { $set('sDurationError', 'Scan the valid duration.'); return; } else { [$hRs, $miNs] = explode('.', $duration) + [0, 0]; // Ensure two parts $hRs = (int) $hRs; $miNs = (int) $miNs; $totalMinutes = $hRs * 60 + $miNs; if ($totalMinutes > 1440) { $set('sDurationError', 'Duration exceeds 24 hours.'); $set('duration', null); $set('end_time', null); return; } $set('sDurationError', null); } }) ->extraAttributes(fn ($get) => [ 'class' => $get('sDurationError') ? 'border-red-500' : '', ]) ->hint(fn ($get) => $get('sDurationError') ? $get('sDurationError') : null) ->hintColor('danger'), Forms\Components\TimePicker::make('end_time') ->required() ->label('End Time') ->readOnly() // ->native(false), ->reactive() ->afterStateUpdated(function ($state, callable $set, callable $get) { $endTime = $get('end_time'); $set('end_time', self::calculateEndTime($get('start_time'), $state)); if (!$endTime) { $set('sEndTimeError', 'Choose the valid start time & duration.'); return; } else { $set('sEndTimeError', null); } }) ->extraAttributes(fn ($get) => [ 'class' => $get('sEndTimeError') ? 'border-red-500' : '', ]) ->hint(fn ($get) => $get('sEndTimeError') ? $get('sEndTimeError') : null) ->hintColor('danger'), Forms\Components\TextInput::make('id') ->hidden() ->readOnly(), ]) ->columns(2), ]); } public static function table(Table $table): Table { return $table ->columns([ Tables\Columns\TextColumn::make('id') ->label('ID') ->numeric() ->sortable(), Tables\Columns\TextColumn::make('name') ->sortable(), Tables\Columns\TextColumn::make('start_time') ->sortable(), Tables\Columns\TextColumn::make('duration') ->sortable(), Tables\Columns\TextColumn::make('end_time'), Tables\Columns\TextColumn::make('block.name') ->sortable(), Tables\Columns\TextColumn::make('plant.name') ->sortable(), Tables\Columns\TextColumn::make('created_at') ->dateTime() ->sortable() ->toggleable(isToggledHiddenByDefault: true), Tables\Columns\TextColumn::make('updated_at') ->dateTime() ->sortable() ->toggleable(isToggledHiddenByDefault: true), Tables\Columns\TextColumn::make('deleted_at') ->dateTime() ->sortable() ->toggleable(isToggledHiddenByDefault: true), ]) ->filters([ Tables\Filters\TrashedFilter::make(), ]) ->actions([ Tables\Actions\ViewAction::make(), Tables\Actions\EditAction::make(), ]) ->bulkActions([ Tables\Actions\BulkActionGroup::make([ Tables\Actions\DeleteBulkAction::make(), Tables\Actions\ForceDeleteBulkAction::make(), Tables\Actions\RestoreBulkAction::make(), ]), ]) ->headerActions([ ImportAction::make() ->importer(ShiftImporter::class), ]); } public static function getRelations(): array { return [ // ]; } public static function getPages(): array { return [ 'index' => Pages\ListShifts::route('/'), 'create' => Pages\CreateShift::route('/create'), 'view' => Pages\ViewShift::route('/{record}'), 'edit' => Pages\EditShift::route('/{record}/edit'), ]; } public static function getEloquentQuery(): Builder { return parent::getEloquentQuery() ->withoutGlobalScopes([ SoftDeletingScope::class, ]); } protected static function calculateEndTime(?string $startTime, ?string $duration): ?string { if (!$startTime || !$duration) { return null; } try { // Convert start_time to Carbon instance $startTimeCarbon = Carbon::createFromFormat('H:i:s', $startTime); // Ensure duration is in a valid numeric format $duration = str_replace(',', '.', $duration); // Handle decimal formats if (!is_numeric($duration)) { return null; // Invalid duration format } // Extract hours and minutes correctly [$hours, $decimalMinutes] = explode('.', $duration) + [0, 0]; // Ensure two parts $hours = (int) $hours; // Convert to integer hours $minutes = (int) $decimalMinutes; // Directly use decimal part as minutes // Calculate end time $endTimeCarbon = $startTimeCarbon->addHours($hours)->addMinutes($minutes); return $endTimeCarbon->format('H:i:s'); // Return formatted end time } catch (\Exception $e) { return null; } } }