openclassify/Modules/Conversation/App/Http/Controllers/ConversationController.php
2026-03-08 21:49:26 +03:00

322 lines
12 KiB
PHP

<?php
namespace Modules\Conversation\App\Http\Controllers;
use App\Http\Controllers\Controller;
use Illuminate\Http\JsonResponse;
use Illuminate\Http\RedirectResponse;
use Illuminate\Http\Request;
use Illuminate\Support\Facades\Schema;
use Illuminate\View\View;
use Modules\Conversation\App\Events\ConversationReadUpdated;
use Modules\Conversation\App\Events\InboxMessageCreated;
use Modules\Conversation\App\Models\Conversation;
use Modules\Conversation\App\Models\ConversationMessage;
use Modules\Conversation\App\Support\QuickMessageCatalog;
use Modules\Listing\Models\Listing;
use Throwable;
class ConversationController extends Controller
{
public function inbox(Request $request): View
{
$user = $request->user();
$userId = $user ? (int) $user->getKey() : null;
$requiresLogin = ! $user;
$messageFilter = $this->resolveMessageFilter($request);
$conversations = collect();
$selectedConversation = null;
if ($userId && $this->messagingTablesReady()) {
try {
[
'conversations' => $conversations,
'selectedConversation' => $selectedConversation,
'markedRead' => $markedRead,
] = $this->resolveInboxState(
$userId,
$messageFilter,
$request->integer('conversation'),
true,
);
if ($selectedConversation && $markedRead) {
broadcast(new ConversationReadUpdated(
$userId,
$selectedConversation->readPayloadFor($userId),
));
}
} catch (Throwable) {
$conversations = collect();
$selectedConversation = null;
}
}
return view('conversation::inbox', [
'conversations' => $conversations,
'selectedConversation' => $selectedConversation,
'messageFilter' => $messageFilter,
'quickMessages' => QuickMessageCatalog::all(),
'requiresLogin' => $requiresLogin,
]);
}
public function state(Request $request): JsonResponse
{
abort_unless($this->messagingTablesReady(), 503, 'Messaging is not available yet.');
$userId = (int) $request->user()->getKey();
$messageFilter = $this->resolveMessageFilter($request);
[
'conversations' => $conversations,
'selectedConversation' => $selectedConversation,
] = $this->resolveInboxState(
$userId,
$messageFilter,
$request->integer('conversation'),
false,
);
return response()->json([
'list_html' => $this->renderInboxList($conversations, $messageFilter, $selectedConversation),
'thread_html' => $this->renderInboxThread($selectedConversation, $messageFilter),
'selected_conversation_id' => $selectedConversation ? (int) $selectedConversation->getKey() : null,
'counts' => [
'unread_messages_total' => Conversation::unreadCountForUser($userId),
],
]);
}
public function start(Request $request, Listing $listing): RedirectResponse | JsonResponse
{
if (! $this->messagingTablesReady()) {
if ($request->expectsJson()) {
return response()->json(['message' => 'Messaging is not available yet.'], 503);
}
return back()->with('error', 'Messaging is not available yet.');
}
$user = $request->user();
if (! $listing->user_id) {
if ($request->expectsJson()) {
return response()->json(['message' => 'A conversation cannot be started for this listing.'], 422);
}
return back()->with('error', 'A conversation cannot be started for this listing.');
}
if ((int) $listing->user_id === (int) $user->getKey()) {
if ($request->expectsJson()) {
return response()->json(['message' => 'You cannot message your own listing.'], 422);
}
return back()->with('error', 'You cannot message your own listing.');
}
$messageBody = trim((string) $request->string('message'));
if ($request->expectsJson() && $messageBody === '') {
return response()->json(['message' => 'Message cannot be empty.'], 422);
}
$conversation = Conversation::openForListingBuyer($listing, (int) $user->getKey());
$user->favoriteListings()->syncWithoutDetaching([$listing->getKey()]);
$message = null;
if ($messageBody !== '') {
$message = $conversation->createMessageFor((int) $user->getKey(), $messageBody);
$this->broadcastMessageCreated($conversation, $message, (int) $user->getKey());
}
if ($request->expectsJson()) {
return $this->conversationJsonResponse($conversation, $message, (int) $user->getKey());
}
return redirect()
->route('panel.inbox.index', array_merge($this->inboxFilters($request), ['conversation' => $conversation->getKey()]))
->with('success', $messageBody !== '' ? 'Message sent.' : 'Conversation started.');
}
public function send(Request $request, Conversation $conversation): RedirectResponse | JsonResponse
{
if (! $this->messagingTablesReady()) {
if ($request->expectsJson()) {
return response()->json(['message' => 'Messaging is not available yet.'], 503);
}
return back()->with('error', 'Messaging is not available yet.');
}
$user = $request->user();
$userId = (int) $user->getKey();
if ((int) $conversation->buyer_id !== $userId && (int) $conversation->seller_id !== $userId) {
abort(403);
}
$payload = $request->validate([
'message' => ['required', 'string', 'max:2000'],
]);
$messageBody = trim($payload['message']);
if ($messageBody === '') {
if ($request->expectsJson()) {
return response()->json(['message' => 'Message cannot be empty.'], 422);
}
return back()->with('error', 'Message cannot be empty.');
}
$message = $conversation->createMessageFor($userId, $messageBody);
$this->broadcastMessageCreated($conversation, $message, $userId);
if ($request->expectsJson()) {
return $this->conversationJsonResponse($conversation, $message, $userId);
}
return redirect()
->route('panel.inbox.index', array_merge($this->inboxFilters($request), ['conversation' => $conversation->getKey()]))
->with('success', 'Message sent.');
}
public function read(Request $request, Conversation $conversation): JsonResponse
{
abort_unless($this->messagingTablesReady(), 503, 'Messaging is not available yet.');
$userId = (int) $request->user()->getKey();
abort_unless($conversation->hasParticipant($userId), 403);
$updated = $conversation->markAsReadFor($userId);
$payload = $conversation->readPayloadFor($userId);
if ($updated > 0) {
broadcast(new ConversationReadUpdated($userId, $payload))->toOthers();
}
return response()->json($payload);
}
private function conversationJsonResponse(Conversation $conversation, ?ConversationMessage $message, int $userId): JsonResponse
{
return response()->json([
'conversation_id' => (int) $conversation->getKey(),
'send_url' => route('conversations.messages.send', $conversation),
'read_url' => route('conversations.read', $conversation),
'conversation' => [
'id' => (int) $conversation->getKey(),
'unread_count' => $conversation->unreadCountForParticipant($userId),
],
'counts' => [
'unread_messages_total' => Conversation::unreadCountForUser($userId),
],
'message' => $message ? $this->messagePayload($message, $userId) : null,
]);
}
private function messagePayload(ConversationMessage $message, int $userId): array
{
return $message->toRealtimePayloadFor($userId);
}
private function inboxFilters(Request $request): array
{
$messageFilter = $this->resolveMessageFilter($request);
return $messageFilter === 'all' ? [] : ['message_filter' => $messageFilter];
}
private function resolveMessageFilter(Request $request): string
{
$messageFilter = (string) $request->string('message_filter', 'all');
return in_array($messageFilter, ['all', 'unread', 'important'], true) ? $messageFilter : 'all';
}
private function resolveInboxState(
int $userId,
string $messageFilter,
?int $conversationId,
bool $markSelectedRead,
): array {
$conversations = Conversation::inboxForUser($userId, $messageFilter);
$selectedConversation = Conversation::resolveSelected($conversations, $conversationId);
$markedRead = false;
if ($selectedConversation) {
$selectedConversation->loadThread();
if ($markSelectedRead) {
$markedRead = $selectedConversation->markAsReadFor($userId) > 0;
$selectedConversation->unread_count = 0;
$conversations = $conversations->map(function (Conversation $conversation) use ($selectedConversation): Conversation {
if ((int) $conversation->getKey() === (int) $selectedConversation->getKey()) {
$conversation->unread_count = 0;
}
return $conversation;
});
}
}
return [
'conversations' => $conversations,
'selectedConversation' => $selectedConversation,
'markedRead' => $markedRead,
];
}
private function renderInboxList($conversations, string $messageFilter, ?Conversation $selectedConversation): string
{
return view('conversation::partials.inbox-list-pane', [
'conversations' => $conversations,
'messageFilter' => $messageFilter,
'selectedConversation' => $selectedConversation,
])->render();
}
private function renderInboxThread(?Conversation $selectedConversation, string $messageFilter): string
{
return view('conversation::partials.inbox-thread-pane', [
'selectedConversation' => $selectedConversation,
'messageFilter' => $messageFilter,
'quickMessages' => QuickMessageCatalog::all(),
])->render();
}
private function broadcastMessageCreated(
Conversation $conversation,
ConversationMessage $message,
int $senderId,
): void {
foreach ($conversation->participantIds() as $participantId) {
$event = new InboxMessageCreated(
$participantId,
$conversation->realtimePayloadFor($participantId, $message),
);
if ($participantId === $senderId) {
broadcast($event)->toOthers();
continue;
}
broadcast($event);
}
}
private function messagingTablesReady(): bool
{
try {
return Schema::hasTable('conversations') && Schema::hasTable('conversation_messages');
} catch (Throwable) {
return false;
}
}
}