mirror of
https://github.com/openclassify/openclassify.git
synced 2026-04-14 11:12:09 -05:00
215 lines
7.5 KiB
PHP
215 lines
7.5 KiB
PHP
<?php
|
||
|
||
namespace Modules\Favorite\App\Http\Controllers;
|
||
|
||
use App\Http\Controllers\Controller;
|
||
use Illuminate\Http\Request;
|
||
use Illuminate\Pagination\LengthAwarePaginator;
|
||
use Illuminate\Support\Facades\Schema;
|
||
use Modules\Category\Models\Category;
|
||
use Modules\Conversation\App\Models\Conversation;
|
||
use Modules\Favorite\App\Models\FavoriteSearch;
|
||
use Modules\Listing\Models\Listing;
|
||
use Modules\User\App\Models\User;
|
||
use Throwable;
|
||
|
||
class FavoriteController extends Controller
|
||
{
|
||
public function index(Request $request)
|
||
{
|
||
$activeTab = (string) $request->string('tab', 'listings');
|
||
if (! in_array($activeTab, ['listings', 'searches', 'sellers'], true)) {
|
||
$activeTab = 'listings';
|
||
}
|
||
|
||
$statusFilter = (string) $request->string('status', 'all');
|
||
if (! in_array($statusFilter, ['all', 'active'], true)) {
|
||
$statusFilter = 'all';
|
||
}
|
||
|
||
$selectedCategoryId = $request->integer('category');
|
||
if ($selectedCategoryId <= 0) {
|
||
$selectedCategoryId = null;
|
||
}
|
||
|
||
$user = $request->user();
|
||
$requiresLogin = ! $user;
|
||
|
||
$categories = collect();
|
||
if ($this->tableExists('categories')) {
|
||
$categories = Category::query()
|
||
->where('is_active', true)
|
||
->orderBy('name')
|
||
->get(['id', 'name']);
|
||
}
|
||
|
||
$favoriteListings = $this->emptyPaginator();
|
||
$favoriteSearches = $this->emptyPaginator();
|
||
$favoriteSellers = $this->emptyPaginator();
|
||
$buyerConversationListingMap = [];
|
||
|
||
if ($user && $activeTab === 'listings') {
|
||
try {
|
||
if ($this->tableExists('favorite_listings')) {
|
||
$favoriteListings = $user->favoriteListings()
|
||
->with(['category:id,name', 'user:id,name'])
|
||
->wherePivot('created_at', '>=', now()->subYear())
|
||
->when($statusFilter === 'active', fn ($query) => $query->where('status', 'active'))
|
||
->when($selectedCategoryId, fn ($query) => $query->where('category_id', $selectedCategoryId))
|
||
->orderByPivot('created_at', 'desc')
|
||
->paginate(10)
|
||
->withQueryString();
|
||
}
|
||
|
||
if (
|
||
$favoriteListings->isNotEmpty()
|
||
&& $this->tableExists('conversations')
|
||
) {
|
||
$userId = (int) $user->getKey();
|
||
$buyerConversationListingMap = Conversation::query()
|
||
->where('buyer_id', $userId)
|
||
->whereIn('listing_id', $favoriteListings->pluck('id')->all())
|
||
->pluck('id', 'listing_id')
|
||
->map(fn ($conversationId) => (int) $conversationId)
|
||
->all();
|
||
}
|
||
} catch (Throwable) {
|
||
$favoriteListings = $this->emptyPaginator();
|
||
$buyerConversationListingMap = [];
|
||
}
|
||
}
|
||
|
||
if ($user && $activeTab === 'searches') {
|
||
try {
|
||
if ($this->tableExists('favorite_searches')) {
|
||
$favoriteSearches = $user->favoriteSearches()
|
||
->with('category:id,name')
|
||
->latest()
|
||
->paginate(10)
|
||
->withQueryString();
|
||
}
|
||
} catch (Throwable) {
|
||
$favoriteSearches = $this->emptyPaginator();
|
||
}
|
||
}
|
||
|
||
if ($user && $activeTab === 'sellers') {
|
||
try {
|
||
if ($this->tableExists('favorite_sellers')) {
|
||
$favoriteSellers = $user->favoriteSellers()
|
||
->withCount([
|
||
'listings as active_listings_count' => fn ($query) => $query->where('status', 'active'),
|
||
])
|
||
->orderByPivot('created_at', 'desc')
|
||
->paginate(10)
|
||
->withQueryString();
|
||
}
|
||
} catch (Throwable) {
|
||
$favoriteSellers = $this->emptyPaginator();
|
||
}
|
||
}
|
||
|
||
return view('favorite::index', [
|
||
'activeTab' => $activeTab,
|
||
'statusFilter' => $statusFilter,
|
||
'selectedCategoryId' => $selectedCategoryId,
|
||
'categories' => $categories,
|
||
'favoriteListings' => $favoriteListings,
|
||
'favoriteSearches' => $favoriteSearches,
|
||
'favoriteSellers' => $favoriteSellers,
|
||
'buyerConversationListingMap' => $buyerConversationListingMap,
|
||
'requiresLogin' => $requiresLogin,
|
||
]);
|
||
}
|
||
|
||
public function toggleListing(Request $request, Listing $listing)
|
||
{
|
||
$isNowFavorite = $request->user()->toggleFavoriteListing($listing);
|
||
|
||
return back()->with('success', $isNowFavorite ? 'İlan favorilere eklendi.' : 'İlan favorilerden kaldırıldı.');
|
||
}
|
||
|
||
public function toggleSeller(Request $request, User $seller)
|
||
{
|
||
$user = $request->user();
|
||
|
||
if ((int) $user->getKey() === (int) $seller->getKey()) {
|
||
return back()->with('error', 'Kendi hesabını favorilere ekleyemezsin.');
|
||
}
|
||
|
||
$isNowFavorite = $user->toggleFavoriteSeller($seller);
|
||
|
||
return back()->with('success', $isNowFavorite ? 'Satıcı favorilere eklendi.' : 'Satıcı favorilerden kaldırıldı.');
|
||
}
|
||
|
||
public function storeSearch(Request $request)
|
||
{
|
||
$data = $request->validate([
|
||
'search' => ['nullable', 'string', 'max:120'],
|
||
'category_id' => ['nullable', 'integer', 'exists:categories,id'],
|
||
]);
|
||
|
||
$filters = FavoriteSearch::normalizeFilters([
|
||
'search' => $data['search'] ?? null,
|
||
'category' => $data['category_id'] ?? null,
|
||
]);
|
||
|
||
if ($filters === []) {
|
||
return back()->with('error', 'Favoriye eklemek için en az bir filtre seçmelisin.');
|
||
}
|
||
|
||
$signature = FavoriteSearch::signatureFor($filters);
|
||
|
||
$categoryName = null;
|
||
if (isset($filters['category'])) {
|
||
$categoryName = Category::query()->whereKey($filters['category'])->value('name');
|
||
}
|
||
|
||
$label = FavoriteSearch::labelFor($filters, is_string($categoryName) ? $categoryName : null);
|
||
|
||
$favoriteSearch = $request->user()->favoriteSearches()->firstOrCreate(
|
||
['signature' => $signature],
|
||
[
|
||
'label' => $label,
|
||
'search_term' => $filters['search'] ?? null,
|
||
'category_id' => $filters['category'] ?? null,
|
||
'filters' => $filters,
|
||
]
|
||
);
|
||
|
||
if (! $favoriteSearch->wasRecentlyCreated) {
|
||
return back()->with('success', 'Bu arama zaten favorilerinde.');
|
||
}
|
||
|
||
return back()->with('success', 'Arama favorilere eklendi.');
|
||
}
|
||
|
||
public function destroySearch(Request $request, FavoriteSearch $favoriteSearch)
|
||
{
|
||
if ((int) $favoriteSearch->user_id !== (int) $request->user()->getKey()) {
|
||
abort(403);
|
||
}
|
||
|
||
$favoriteSearch->delete();
|
||
|
||
return back()->with('success', 'Favori arama silindi.');
|
||
}
|
||
|
||
private function tableExists(string $table): bool
|
||
{
|
||
try {
|
||
return Schema::hasTable($table);
|
||
} catch (Throwable) {
|
||
return false;
|
||
}
|
||
}
|
||
|
||
private function emptyPaginator(): LengthAwarePaginator
|
||
{
|
||
return new LengthAwarePaginator([], 0, 10, 1, [
|
||
'path' => request()->url(),
|
||
'query' => request()->query(),
|
||
]);
|
||
}
|
||
}
|