Files
fuel-price/app/Filament/Resources/Searches/SearchResource.php
Ovidiu U 7101ed3550
Some checks failed
linter / quality (push) Has been cancelled
tests / ci (8.3) (push) Has been cancelled
tests / ci (8.4) (push) Has been cancelled
tests / ci (8.5) (push) Has been cancelled
feat: add postcode resolution to /api/stations and Filament SearchResource
Extends NearbyStationsRequest to accept `postcode` (full or outcode) as an alternative to lat/lng. PostcodeService resolves it via postcodes.io and falls through to coordinates. Also adds SearchResource to the Filament admin panel for viewing logged search activity with fuel type filter and price/distance stats columns. Includes SQLite GREATEST/LEAST function polyfills in AppServiceProvider for test compatibility.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-05 19:10:25 +01:00

82 lines
2.9 KiB
PHP

<?php
namespace App\Filament\Resources\Searches;
use App\Filament\NavigationGroup;
use App\Filament\Resources\Searches\Pages\ListSearches;
use App\Models\Search;
use Filament\Resources\Resource;
use Filament\Tables\Columns\TextColumn;
use Filament\Tables\Filters\SelectFilter;
use Filament\Tables\Table;
class SearchResource extends Resource
{
protected static ?string $model = Search::class;
protected static string|\UnitEnum|null $navigationGroup = NavigationGroup::Data;
protected static ?string $navigationLabel = 'Searches';
protected static ?int $navigationSort = 4;
public static function table(Table $table): Table
{
return $table
->columns([
TextColumn::make('searched_at')
->label('Searched At')
->dateTime('d M Y H:i')
->sortable(),
TextColumn::make('fuel_type')
->label('Fuel Type')
->badge(),
TextColumn::make('results_count')
->label('Results')
->numeric()
->sortable(),
TextColumn::make('lowest_pence')
->label('Lowest')
->formatStateUsing(fn (int $state): string => number_format($state / 100, 1).'p')
->sortable(),
TextColumn::make('highest_pence')
->label('Highest')
->formatStateUsing(fn (int $state): string => number_format($state / 100, 1).'p')
->sortable(),
TextColumn::make('avg_pence')
->label('Average')
->formatStateUsing(fn (string $state): string => number_format((float) $state / 100, 1).'p')
->sortable(),
TextColumn::make('lat_bucket')
->label('Area (lat/lng)')
->formatStateUsing(fn (Search $record): string => $record->lat_bucket.', '.$record->lng_bucket)
->toggleable(isToggledHiddenByDefault: true),
TextColumn::make('ip_hash')
->label('IP Hash')
->limit(16)
->toggleable(isToggledHiddenByDefault: true),
])
->defaultSort('searched_at', 'desc')
->filters([
SelectFilter::make('fuel_type')
->options([
'E10' => 'E10',
'E5' => 'E5',
'B7_STANDARD' => 'B7 Standard',
'B7_PREMIUM' => 'B7 Premium',
'B10' => 'B10',
'HVO' => 'HVO',
]),
])
->recordActions([])
->toolbarActions([]);
}
public static function getPages(): array
{
return [
'index' => ListSearches::route('/'),
];
}
}