- Create useFetchProjects composable for API integration - Implement responsive grid layout (1/2/3/4 columns) - Add stagger fadeInUp animation with prefers-reduced-motion support - Include loading skeleton, error state with retry, and empty state - Configure SEO meta tags via useSeo composable - Update Project model ordered scope for proper sorting Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
53 lines
1.2 KiB
PHP
53 lines
1.2 KiB
PHP
<?php
|
|
|
|
namespace App\Models;
|
|
|
|
use App\Traits\HasTranslations;
|
|
use Illuminate\Database\Eloquent\Model;
|
|
use Illuminate\Database\Eloquent\Builder;
|
|
use Illuminate\Database\Eloquent\Relations\BelongsToMany;
|
|
|
|
class Project extends Model
|
|
{
|
|
use HasTranslations;
|
|
protected $fillable = [
|
|
'slug',
|
|
'title_key',
|
|
'description_key',
|
|
'short_description_key',
|
|
'image',
|
|
'url',
|
|
'github_url',
|
|
'date_completed',
|
|
'is_featured',
|
|
'display_order',
|
|
];
|
|
|
|
protected function casts(): array
|
|
{
|
|
return [
|
|
'date_completed' => 'date',
|
|
'is_featured' => 'boolean',
|
|
];
|
|
}
|
|
|
|
public function skills(): BelongsToMany
|
|
{
|
|
return $this->belongsToMany(Skill::class, 'skill_project')
|
|
->withPivot(['level_before', 'level_after', 'level_description_key'])
|
|
->withTimestamps();
|
|
}
|
|
|
|
public function scopeFeatured(Builder $query): Builder
|
|
{
|
|
return $query->where('is_featured', true);
|
|
}
|
|
|
|
public function scopeOrdered(Builder $query): Builder
|
|
{
|
|
return $query->orderByDesc('is_featured')
|
|
->orderByDesc('date_completed')
|
|
->orderBy('display_order');
|
|
}
|
|
}
|