Proyectos de Subversion LeadersLinked - Services

Rev

Rev 673 | Rev 676 | Ir a la última revisión | | Comparar con el anterior | Ultima modificación | Ver Log |

Rev Autor Línea Nro. Línea
1 efrain 1
<?php
2
 
3
declare(strict_types=1);
4
 
5
namespace LeadersLinked\Controller;
6
 
7
use Laminas\Db\Adapter\AdapterInterface;
8
 
9
use Laminas\Mvc\Controller\AbstractActionController;
10
use Laminas\Log\LoggerInterface;
11
use Laminas\View\Model\ViewModel;
12
 
13
use LeadersLinked\Library\Functions;
14
use LeadersLinked\Mapper\UserMapper;
15
 
16
use LeadersLinked\Model\User;
17
 
18
use LeadersLinked\Mapper\UserBlockedMapper;
19
 
20
use LeadersLinked\Mapper\CompanyMapper;
21
use LeadersLinked\Mapper\CompanyUserMapper;
307 www 22
use LeadersLinked\Mapper\ConversationMapper;
23
use LeadersLinked\Mapper\MessageMapper;
24
use LeadersLinked\Model\Conversation;
1 efrain 25
use LeadersLinked\Library\Image;
26
use Laminas\View\Model\JsonModel;
307 www 27
use LeadersLinked\Form\InMail\SendForm;
28
use LeadersLinked\Model\Message;
29
use LeadersLinked\Model\VideoConvert;
30
use LeadersLinked\Mapper\VideoConvertMapper;
1 efrain 31
use Laminas\Mvc\I18n\Translator;
199 efrain 32
use LeadersLinked\Mapper\AbuseReportMapper;
33
use LeadersLinked\Model\AbuseReport;
345 www 34
use LeadersLinked\Library\Storage;
1 efrain 35
 
36
class InMailController extends AbstractActionController
37
{
38
    /**
39
     *
40
     * @var \Laminas\Db\Adapter\AdapterInterface
41
     */
42
    private $adapter;
582 ariadna 43
 
1 efrain 44
    /**
45
     *
46
     * @var \LeadersLinked\Cache\CacheInterface
47
     */
48
    private $cache;
582 ariadna 49
 
50
 
1 efrain 51
    /**
52
     *
53
     * @var \Laminas\Log\LoggerInterface
54
     */
55
    private $logger;
582 ariadna 56
 
1 efrain 57
    /**
58
     *
59
     * @var array
60
     */
61
    private $config;
582 ariadna 62
 
63
 
1 efrain 64
    /**
65
     *
66
     * @var \Laminas\Mvc\I18n\Translator
67
     */
68
    private $translator;
582 ariadna 69
 
70
 
1 efrain 71
    /**
72
     *
73
     * @param \Laminas\Db\Adapter\AdapterInterface $adapter
74
     * @param \LeadersLinked\Cache\CacheInterface $cache
75
     * @param \Laminas\Log\LoggerInterface LoggerInterface $logger
76
     * @param array $config
77
     * @param \Laminas\Mvc\I18n\Translator $translator
78
     */
79
    public function __construct($adapter, $cache, $logger, $config, $translator)
80
    {
81
        $this->adapter      = $adapter;
82
        $this->cache        = $cache;
83
        $this->logger       = $logger;
84
        $this->config       = $config;
85
        $this->translator   = $translator;
86
    }
87
 
88
    /**
667 stevensc 89
     * Genera el listado de conversaciones del usuario actual
1 efrain 90
     *
667 stevensc 91
     * Endpoint: GET /inmail
1 efrain 92
     *
667 stevensc 93
     * Parámetros de ruta:
94
     * - id: UUID del usuario o empresa para seleccionar automáticamente (opcional)
95
     *
96
     * Respuesta exitosa:
1 efrain 97
     * [
667 stevensc 98
     *   success: true,
99
     *   data: [
100
     *     [
101
     *       uuid: UUID del usuario/empresa con quien se tiene la conversación,
102
     *       name: nombre del usuario/empresa,
103
     *       image: URL de la imagen del usuario/empresa,
104
     *       profile: URL del perfil del usuario/empresa,
105
     *       last_message: tiempo transcurrido del último mensaje,
307 www 106
     *       count_unread: cantidad de mensajes sin leer,
667 stevensc 107
     *       messages_link: URL para recuperar los mensajes,
108
     *       send_link: URL para enviar mensaje,
109
     *       selected: 0 = no seleccionado, 1 = seleccionado,
110
     *       delete_link: URL para eliminar la conversación
307 www 111
     *     ]
667 stevensc 112
     *   ]
113
     * ]
1 efrain 114
     *
667 stevensc 115
     * Respuesta de error:
116
     * [
117
     *   success: false,
118
     *   data: mensaje de error
119
     * ]
1 efrain 120
     *
121
     * @return \Laminas\View\Model\JsonModel
122
     */
123
    public function indexAction()
124
    {
125
        $request = $this->getRequest();
667 stevensc 126
 
127
        // Verificar que la petición sea GET
128
        if (!$request->isGet()) {
129
            return new JsonModel([
130
                'success' => false,
131
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
132
            ]);
133
        }
307 www 134
 
667 stevensc 135
        // ========================================
136
        // INICIALIZACIÓN Y OBTENCIÓN DE DATOS BÁSICOS
137
        // ========================================
138
 
139
        // Obtener el usuario actual autenticado
140
        $currentUserPlugin = $this->plugin('currentUserPlugin');
141
        $currentUser = $currentUserPlugin->getUser();
307 www 142
 
667 stevensc 143
        // Obtener la fecha actual de la base de datos para cálculos de tiempo
144
        $userMapper = UserMapper::getInstance($this->adapter);
145
        $now = $userMapper->getDatebaseNow();
1 efrain 146
 
667 stevensc 147
        // ========================================
148
        // OBTENCIÓN DE USUARIOS BLOQUEADOS
149
        // ========================================
150
 
151
        // Obtener IDs de usuarios que el usuario actual ha bloqueado
152
        $userBlockedMapper = UserBlockedMapper::getInstance($this->adapter);
153
        $user_blocked_ids = $userBlockedMapper->fetchAllBlockedReturnIds($currentUser->id);
199 efrain 154
 
667 stevensc 155
        // Obtener IDs de usuarios que han bloqueado al usuario actual
156
        $user_blocked_me_ids = $userBlockedMapper->fetchAllUserBlockMeReturnIds($currentUser->id);
1 efrain 157
 
667 stevensc 158
        // ========================================
159
        // INICIALIZACIÓN DE MAPPER Y VARIABLES
160
        // ========================================
161
 
162
        $conversationMapper = ConversationMapper::getInstance($this->adapter);
163
        $messageMapper = MessageMapper::getInstance($this->adapter);
164
        $conversations = [];
165
        $selected_user_id = 0;
307 www 166
 
667 stevensc 167
        // ========================================
168
        // PROCESAMIENTO DE USUARIO/EMPRESA SELECCIONADO
169
        // ========================================
170
 
171
        $id = $this->params()->fromRoute('id');
172
        if ($id) {
173
            // ========================================
174
            // BÚSQUEDA COMO EMPRESA PRIMERO
175
            // ========================================
176
 
177
            $companyMapper = CompanyMapper::getInstance($this->adapter);
178
            $company = $companyMapper->fetchOneByUuid($id);
179
 
180
            if ($company) {
181
                // Buscar el propietario de la empresa
182
                $companyUserMapper = CompanyUserMapper::getInstance($this->adapter);
183
                $companyUser = $companyUserMapper->fetchOwnerByCompanyId($company->id);
307 www 184
 
667 stevensc 185
                // Verificar que no haya bloqueos entre usuarios
186
                if (!in_array($companyUser->user_id, $user_blocked_ids) &&
187
                    !in_array($companyUser->id, $user_blocked_me_ids)) {
188
 
189
                    $user = $userMapper->fetchOne($companyUser->user_id);
190
                    $selected_user_id = $user->id;
191
 
192
                    // Buscar conversación existente
193
                    $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
307 www 194
 
667 stevensc 195
                    // Calcular tiempo del último mensaje y mensajes no leídos
196
                    $timeElapsedString = '';
197
                    if ($conversation) {
198
                        $unread = $messageMapper->fetchCountUnreadMessagesByConversationIdAndReceiverId(
199
                            $conversation->id,
200
                            $currentUser->id
201
                        );
202
                        $lastMessage = $messageMapper->fetchLastUnreadMessagesByConversationIdAndReceiverId(
203
                            $conversation->id,
204
                            $currentUser->id
205
                        );
582 ariadna 206
 
667 stevensc 207
                        if ($lastMessage) {
208
                            $timeElapsedString = $this->timeAgo($lastMessage, $now);
209
                        }
210
                    } else {
211
                        $unread = 0;
212
                    }
582 ariadna 213
 
667 stevensc 214
                    // Agregar conversación de empresa a la lista
215
                    array_push($conversations, [
216
                        'uuid' => $company->uuid,
217
                        'name' => $company->name,
218
                        'image' => $this->url()->fromRoute('storage', [
219
                            'type' => 'company',
220
                            'code' => $company->uuid,
221
                            'filename' => $company->image
222
                        ], ['force_canonical' => true]),
223
                        'profile' => $this->url()->fromRoute('company/view', ['id' => $company->uuid]),
224
                        'last_message' => $timeElapsedString,
225
                        'count_unread' => $unread,
675 stevensc 226
                        'messages_link' => $this->url()->fromRoute('inmail/messages', ['id' => $company->uuid]),
673 stevensc 227
                        'send_link' => $this->url()->fromRoute('inmail/message/send', ['id' => $company->uuid]),
667 stevensc 228
                        'selected' => 1, // Marcar como seleccionado
673 stevensc 229
                        'delete_link' => $this->url()->fromRoute('inmail/message/delete', ['id' => $company->uuid]),
667 stevensc 230
                    ]);
231
                }
232
            }
233
            // ========================================
234
            // BÚSQUEDA COMO USUARIO INDIVIDUAL
235
            // ========================================
236
            else {
237
                $user = $userMapper->fetchOneByUuid($id);
238
                if ($user) {
239
                    // Verificar que no haya bloqueos entre usuarios
240
                    if (!in_array($user->id, $user_blocked_ids) &&
241
                        !in_array($user->id, $user_blocked_me_ids)) {
582 ariadna 242
 
667 stevensc 243
                        $timeElapsedString = '';
582 ariadna 244
                        $selected_user_id = $user->id;
667 stevensc 245
 
246
                        // Buscar conversación existente
582 ariadna 247
                        $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
667 stevensc 248
 
582 ariadna 249
                        if ($conversation) {
667 stevensc 250
                            $unread = $messageMapper->fetchCountUnreadMessagesByConversationIdAndReceiverId(
251
                                $conversation->id,
252
                                $currentUser->id
253
                            );
254
                            $lastMessage = $messageMapper->fetchLastUnreadMessagesByConversationIdAndReceiverId(
255
                                $conversation->id,
256
                                $currentUser->id
257
                            );
258
 
582 ariadna 259
                            if ($lastMessage) {
667 stevensc 260
                                $timeElapsedString = $this->timeAgo($lastMessage, $now);
582 ariadna 261
                            }
262
                        } else {
263
                            $unread = 0;
264
                        }
265
 
667 stevensc 266
                        // Agregar conversación de usuario a la lista
582 ariadna 267
                        array_push($conversations, [
667 stevensc 268
                            'uuid' => $user->uuid,
269
                            'name' => trim($user->first_name . ' ' . $user->last_name),
270
                            'image' => $this->url()->fromRoute('storage', [
271
                                'type' => 'user',
272
                                'code' => $user->uuid,
273
                                'filename' => $user->image
274
                            ], ['force_canonical' => true]),
275
                            'profile' => $this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
582 ariadna 276
                            'last_message' => $timeElapsedString,
277
                            'count_unread' => $unread,
675 stevensc 278
                            'messages_link' => $this->url()->fromRoute('inmail/messages', ['id' => $user->uuid]),
582 ariadna 279
                            'send_link' => $this->url()->fromRoute('inmail/message/send', ['id' => $user->uuid]),
667 stevensc 280
                            'selected' => 1, // Marcar como seleccionado
281
                            'delete_link' => $this->url()->fromRoute('inmail/message/delete', ['id' => $user->uuid]),
582 ariadna 282
                        ]);
283
                    }
284
                }
285
            }
667 stevensc 286
        }
287
 
288
        $records = $conversationMapper->fetchAllByUserId($currentUser->id);
289
 
673 stevensc 290
        foreach ($records as $record) {
667 stevensc 291
            // Omitir conversaciones que ya fueron procesadas como seleccionadas
292
            if ($selected_user_id) {
293
                if ($record->sender_id == $currentUser->id && $record->receiver_id == $selected_user_id) {
294
                    continue;
1 efrain 295
                }
667 stevensc 296
                if ($record->receiver_id == $currentUser->id && $record->sender_id == $selected_user_id) {
297
                    continue;
582 ariadna 298
                }
667 stevensc 299
            }
1 efrain 300
 
667 stevensc 301
            // ========================================
302
            // DETERMINACIÓN DEL OTRO USUARIO EN LA CONVERSACIÓN
303
            // ========================================
304
 
305
            if ($record->sender_id == $currentUser->id) {
306
                $user = $userMapper->fetchOne($record->receiver_id);
307
            }
308
            if ($record->receiver_id == $currentUser->id) {
309
                $user = $userMapper->fetchOne($record->sender_id);
310
 
311
                // Omitir usuarios bloqueados
312
                if (in_array($user->id, $user_blocked_ids) || in_array($user->id, $user_blocked_me_ids)) {
313
                    continue;
582 ariadna 314
                }
667 stevensc 315
            }
1 efrain 316
 
667 stevensc 317
            // ========================================
318
            // CÁLCULO DE TIEMPO Y MENSAJES NO LEÍDOS
319
            // ========================================
320
 
321
            $timeElapsedString = '';
322
            $unread = $messageMapper->fetchCountUnreadMessagesByConversationIdAndReceiverId(
323
                $record->id,
324
                $currentUser->id
325
            );
326
            $lastMessage = $messageMapper->fetchLastMessagesByConversationIdAndReceiverId(
327
                $record->id,
328
                $currentUser->id
329
            );
1 efrain 330
 
667 stevensc 331
            if ($lastMessage) {
332
                $timeElapsedString = $this->timeAgo($lastMessage, $now);
582 ariadna 333
            }
1 efrain 334
 
667 stevensc 335
            // ========================================
336
            // AGREGAR CONVERSACIÓN A LA LISTA
337
            // ========================================
338
 
339
            array_push($conversations, [
340
                'uuid' => $user->uuid,
341
                'name' => trim($user->first_name . ' ' . $user->last_name),
342
                'image' => $this->url()->fromRoute('storage', [
343
                    'type' => 'user',
344
                    'code' => $user->uuid,
345
                    'filename' => $user->image
346
                ], ['force_canonical' => true]),
347
                'profile' => $this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
348
                'last_message' => $timeElapsedString,
349
                'count_unread' => $unread,
675 stevensc 350
                'messages_link' => $this->url()->fromRoute('inmail/messages', ['id' => $user->uuid]),
667 stevensc 351
                'send_link' => $this->url()->fromRoute('inmail/message/send', ['id' => $user->uuid]),
352
                'selected' => 0, // No seleccionado
353
                'delete_link' => $this->url()->fromRoute('inmail/message/delete', ['id' => $user->uuid]),
354
            ]);
355
        }
1 efrain 356
 
667 stevensc 357
        // ========================================
358
        // ORDENAMIENTO DE CONVERSACIONES
359
        // ========================================
360
 
361
        // Ordenar por: seleccionados primero, luego por fecha del último mensaje
362
        usort($conversations, function ($a, $b) {
363
            if ($a['selected'] == $b['selected']) {
364
                if ($a['last_message'] == $b['last_message']) {
365
                    return 0;
582 ariadna 366
                } else {
667 stevensc 367
                    return $a['last_message'] < $b['last_message'] ? -1 : 1;
582 ariadna 368
                }
667 stevensc 369
            } else {
370
                return $a['selected'] < $b['selected'] ? -1 : 1;
371
            }
372
        });
1 efrain 373
 
667 stevensc 374
        // ========================================
375
        // RESPUESTA FINAL
376
        // ========================================
377
 
378
        return new JsonModel([
379
            'success' => true,
380
            'data' => $conversations
381
        ]);
1 efrain 382
    }
307 www 383
 
384
    /**
666 stevensc 385
     * Recupera los mensajes de una conversación específica entre usuarios
386
     *
675 stevensc 387
     * Endpoint: GET /inmail/messages/:id
666 stevensc 388
     *
389
     * Parámetros de consulta:
390
     * - page: número de página para paginación (opcional, por defecto 0)
391
     *
392
     * Respuesta exitosa:
307 www 393
     * [
394
     *   success: true,
666 stevensc 395
     *   data: [
396
     *     [
397
     *       uuid: UUID del mensaje,
398
     *       sender_name: nombre del remitente,
399
     *       sender_image: URL de la imagen del remitente,
400
     *       sender_profile: URL del perfil del remitente,
401
     *       receiver_name: nombre del destinatario,
402
     *       receiver_image: URL de la imagen del destinatario,
403
     *       receiver_profile: URL del perfil del destinatario,
404
     *       side: 'left' (mensaje propio) | 'right' (mensaje recibido),
405
     *       message: contenido del mensaje,
406
     *       type: 'text' | 'image' | 'video' | 'document',
407
     *       filename: URL del archivo adjunto (si existe),
408
     *       date: tiempo transcurrido desde el envío,
409
     *       url_abuse_report: URL para reportar abuso (solo mensajes recibidos)
410
     *     ]
411
     *   ],
412
     *   pagination: información de paginación
413
     * ]
414
     *
415
     * Respuesta de error:
307 www 416
     * [
666 stevensc 417
     *   success: false,
418
     *   data: mensaje de error
307 www 419
     * ]
666 stevensc 420
     *
307 www 421
     * @return \Laminas\View\Model\JsonModel
422
     */
675 stevensc 423
    public function messagesAction()
1 efrain 424
    {
425
        $request = $this->getRequest();
666 stevensc 426
 
427
        // Verificar que la petición sea GET
428
        if (!$request->isGet()) {
429
            return new JsonModel([
430
                'success' => false,
431
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
432
            ]);
433
        }
307 www 434
 
666 stevensc 435
        // ========================================
436
        // INICIALIZACIÓN Y VALIDACIÓN DE PARÁMETROS
437
        // ========================================
438
 
439
        // Obtener parámetros de la petición
440
        $page = intval($this->params()->fromQuery('page', 0), 10);
441
        $id = $this->params()->fromRoute('id');
442
 
443
        // Validar que se proporcione el ID del usuario
444
        if (!$id) {
445
            return new JsonModel([
446
                'success' => false,
447
                'data' => 'ERROR_PARAMETERS_ARE_INVALID'
448
            ]);
449
        }
307 www 450
 
666 stevensc 451
        // Obtener el usuario actual autenticado
452
        $currentUserPlugin = $this->plugin('currentUserPlugin');
453
        $currentUser = $currentUserPlugin->getUser();
582 ariadna 454
 
666 stevensc 455
        // Obtener la fecha actual de la base de datos para cálculos de tiempo
456
        $userMapper = UserMapper::getInstance($this->adapter);
457
        $now = $userMapper->getDatebaseNow();
307 www 458
 
666 stevensc 459
        // ========================================
460
        // VALIDACIÓN DEL USUARIO DESTINATARIO
461
        // ========================================
462
 
463
        // Buscar el usuario destinatario por UUID
464
        $user = $userMapper->fetchOneByUuid($id);
465
        if (!$user) {
466
            return new JsonModel([
467
                'success' => false,
468
                'data' => 'ERROR_REQUEST_IS_INVALID'
469
            ]);
470
        }
307 www 471
 
666 stevensc 472
        // ========================================
473
        // OBTENCIÓN DE REPORTES DE ABUSO
474
        // ========================================
475
 
476
        // Obtener IDs de mensajes reportados por el usuario actual
477
        $abuseReportMapper = AbuseReportMapper::getInstance($this->adapter);
478
        $abuse_report_message_ids = $abuseReportMapper->fetchAllDataByUserReportingIdAndTypeReturnIds(
479
            $currentUser->id,
480
            AbuseReport::TYPE_INMAIL_MESSAGE
481
        );
307 www 482
 
666 stevensc 483
        // ========================================
484
        // BÚSQUEDA DE LA CONVERSACIÓN
485
        // ========================================
486
 
487
        $conversationMapper = ConversationMapper::getInstance($this->adapter);
488
        $messageMapper = MessageMapper::getInstance($this->adapter);
489
 
490
        // Buscar la conversación entre los dos usuarios
491
        $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
492
 
493
        $messages = [];
494
 
495
        // ========================================
496
        // PROCESAMIENTO DE MENSAJES
497
        // ========================================
498
 
499
        if ($conversation) {
500
            // Obtener mensajes paginados de la conversación
501
            $records = $messageMapper->getAllMessagesPaginatorByConversationId($conversation->id, $page);
582 ariadna 502
 
666 stevensc 503
            foreach ($records as $record) {
504
                // Calcular tiempo transcurrido desde el envío del mensaje
505
                $timeElapsedString = $this->timeAgo($record->added_on, $now);
307 www 506
 
666 stevensc 507
                // ========================================
508
                // MENSAJES ENVIADOS POR EL USUARIO ACTUAL
509
                // ========================================
510
 
511
                if ($record->sender_id == $currentUser->id) {
512
                    array_push($messages, [
513
                        'uuid' => $record->uuid,
514
                        'sender_name' => trim($currentUser->first_name . ' ' . $currentUser->last_name),
515
                        'sender_image' => $this->url()->fromRoute('storage', [
516
                            'type' => 'user',
517
                            'filename' => $currentUser->image,
518
                            'code' => $currentUser->uuid
519
                        ], ['force_canonical' => true]),
520
                        'sender_profile' => $this->url()->fromRoute('profile/view', ['id' => $currentUser->uuid]),
521
                        'receiver_name' => trim($user->first_name . ' ' . $user->last_name),
522
                        'receiver_image' => $this->url()->fromRoute('storage', [
523
                            'type' => 'user',
524
                            'filename' => $user->image,
525
                            'code' => $user->uuid
526
                        ], ['force_canonical' => true]),
527
                        'receiver_profile' => $this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
528
                        'side' => 'left', // Mensaje propio (lado izquierdo)
529
                        'message' => $record->message,
530
                        'type' => $record->type,
531
                        'filename' => $record->filename ? $this->url()->fromRoute('storage', [
532
                            'type' => 'message',
533
                            'filename' => $record->filename,
534
                            'code' => $record->uuid
535
                        ], ['force_canonical' => true]) : '',
536
                        'date' => $timeElapsedString,
537
                        'url_abuse_report' => '', // No se puede reportar mensajes propios
538
                    ]);
539
                }
540
                // ========================================
541
                // MENSAJES RECIBIDOS DEL OTRO USUARIO
542
                // ========================================
543
                else {
544
                    // Verificar si el mensaje ha sido reportado por abuso
545
                    if (in_array($record->id, $abuse_report_message_ids)) {
546
                        // Mensaje reportado - mostrar mensaje de contenido reportado
547
                        array_push($messages, [
548
                            'uuid' => $record->uuid,
549
                            'sender_name' => trim($user->first_name . ' ' . $user->last_name),
550
                            'sender_image' => $this->url()->fromRoute('storage', [
551
                                'type' => 'user',
552
                                'filename' => $user->image,
553
                                'code' => $user->uuid
554
                            ], ['force_canonical' => true]),
555
                            'sender_profile' => $this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
556
                            'receiver_name' => trim($currentUser->first_name . ' ' . $currentUser->last_name),
557
                            'receiver_image' => $this->url()->fromRoute('storage', [
558
                                'type' => 'user',
559
                                'filename' => $currentUser->image,
560
                                'code' => $user->uuid
561
                            ], ['force_canonical' => true]),
562
                            'receiver_profile' => $this->url()->fromRoute('profile/view', ['id' => $currentUser->uuid]),
563
                            'side' => 'right', // Mensaje recibido (lado derecho)
564
                            'message' => 'LABEL_ABUSE_CONTENT_REPORTED',
565
                            'type' => Message::TYPE_TEXT,
566
                            'filename' => '',
567
                            'date' => $timeElapsedString,
568
                            'url_abuse_report' => '', // Ya no se puede reportar
569
                        ]);
570
                    } else {
571
                        // Mensaje normal - generar URL para reportar abuso
572
                        $url_abuse_report = $this->url()->fromRoute('helpers/abuse-report', [
573
                            'type' => 'message',
574
                            'id' => $record->uuid
575
                        ], ['force_canonical' => true]);
582 ariadna 576
 
307 www 577
                        array_push($messages, [
578
                            'uuid' => $record->uuid,
666 stevensc 579
                            'sender_name' => trim($user->first_name . ' ' . $user->last_name),
580
                            'sender_image' => $this->url()->fromRoute('storage', [
581
                                'type' => 'user',
582
                                'filename' => $user->image,
583
                                'code' => $user->uuid
584
                            ], ['force_canonical' => true]),
585
                            'sender_profile' => $this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
586
                            'receiver_name' => trim($currentUser->first_name . ' ' . $currentUser->last_name),
587
                            'receiver_image' => $this->url()->fromRoute('storage', [
588
                                'type' => 'user',
589
                                'filename' => $currentUser->image,
590
                                'code' => $user->uuid
591
                            ], ['force_canonical' => true]),
592
                            'receiver_profile' => $this->url()->fromRoute('profile/view', ['id' => $currentUser->uuid]),
593
                            'side' => 'right', // Mensaje recibido (lado derecho)
307 www 594
                            'message' => $record->message,
595
                            'type' => $record->type,
666 stevensc 596
                            'filename' => $record->filename ? $this->url()->fromRoute('storage', [
597
                                'type' => 'message',
598
                                'filename' => $record->filename,
599
                                'code' => $record->uuid
600
                            ], ['force_canonical' => true]) : '',
307 www 601
                            'date' => $timeElapsedString,
666 stevensc 602
                            'url_abuse_report' => $url_abuse_report,
307 www 603
                        ]);
582 ariadna 604
 
666 stevensc 605
                        // Marcar el mensaje como leído
307 www 606
                        $messageMapper->markAsRead($record->id);
261 efrain 607
                    }
307 www 608
                }
1 efrain 609
            }
307 www 610
 
666 stevensc 611
            // ========================================
612
            // RESPUESTA CON MENSAJES Y PAGINACIÓN
613
            // ========================================
614
 
307 www 615
            return new JsonModel([
616
                'success' => true,
617
                'data' => $messages,
666 stevensc 618
                'pagination' => $records->getPages()
307 www 619
            ]);
1 efrain 620
        }
666 stevensc 621
 
622
        // ========================================
623
        // RESPUESTA CUANDO NO EXISTE CONVERSACIÓN
624
        // ========================================
625
 
626
        return new JsonModel([
627
            'success' => true,
628
            'data' => $messages,
629
            'pagination' => 1
630
        ]);
1 efrain 631
    }
307 www 632
 
633
    /**
589 ariadna 634
     * Envía un mensaje InMail a un usuario específico
307 www 635
     *
589 ariadna 636
     * Endpoint: POST /inmail/message/send/:id
637
     *
638
     * Parámetros requeridos:
639
     * - message: texto plano del mensaje
640
     *
641
     * Parámetros opcionales:
642
     * - file: archivo adjunto (jpg, jpeg, png, mp4, webm, mpeg, pdf)
643
     *
644
     * @return JsonModel
307 www 645
     */
646
    public function sendMessageAction()
1 efrain 647
    {
589 ariadna 648
        // Verificar que la petición sea POST
1 efrain 649
        $request = $this->getRequest();
589 ariadna 650
        if (!$request->isPost()) {
651
            return new JsonModel([
652
                'success' => false,
653
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
654
            ]);
655
        }
307 www 656
 
589 ariadna 657
        // Obtener el usuario actual
658
        $currentUserPlugin = $this->plugin('currentUserPlugin');
659
        $currentUser = $currentUserPlugin->getUser();
307 www 660
 
589 ariadna 661
        // Obtener la fecha actual de la base de datos
662
        $userMapper = UserMapper::getInstance($this->adapter);
663
        $now = $userMapper->getDatebaseNow();
582 ariadna 664
 
589 ariadna 665
        // Validar el ID del destinatario
666
        $id = $this->params()->fromRoute('id');
667
        if (!$id) {
668
            return new JsonModel([
669
                'success' => false,
670
                'data' => 'ERROR_PARAMETERS_ARE_INVALID'
671
            ]);
672
        }
307 www 673
 
589 ariadna 674
        // Buscar el usuario destinatario
675
        $user = $userMapper->fetchOneByUuidAndNetworkId($id, $currentUser->network_id);
676
        if (!$user) {
677
            return new JsonModel([
678
                'success' => false,
679
                'data' => 'ERROR_REQUEST_IS_INVALID'
680
            ]);
681
        }
307 www 682
 
589 ariadna 683
        // Combinar datos POST y archivos
684
        $data = array_merge($request->getPost()->toArray(), $request->getFiles()->toArray());
307 www 685
 
589 ariadna 686
        // Si se envía un archivo sin error, usar su nombre como mensaje
687
        if (!empty($data['file']) && empty($data['file']['error'])) {
688
            $data['message'] = $data['file']['name'];
689
        }
307 www 690
 
589 ariadna 691
        // Validar los datos del formulario
692
        $form = new SendForm();
693
        $form->setData($data);
582 ariadna 694
 
589 ariadna 695
        if (!$form->isValid()) {
696
            // Si el formulario no es válido, devolver los errores
697
            $messages = [];
698
            $form_messages = (array) $form->getMessages();
699
            foreach ($form_messages as $fieldname => $field_messages) {
700
                $messages[$fieldname] = array_values($field_messages);
307 www 701
            }
589 ariadna 702
            return new JsonModel([
703
                'success' => false,
704
                'data' => $messages
705
            ]);
706
        }
582 ariadna 707
 
589 ariadna 708
        // Obtener los datos validados
709
        $dataPost = (array) $form->getData();
307 www 710
 
589 ariadna 711
        // Buscar o crear la conversación
712
        $conversationMapper = ConversationMapper::getInstance($this->adapter);
713
        $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
307 www 714
 
589 ariadna 715
        if ($conversation) {
716
            // Actualizar estado de la conversación existente
717
            $conversation->receiver_status = Conversation::STATUS_NORMAL;
718
            $conversation->sender_status = Conversation::STATUS_NORMAL;
307 www 719
 
589 ariadna 720
            if (!$conversationMapper->update($conversation)) {
721
                return new JsonModel([
722
                    'success' => false,
723
                    'data' => $conversationMapper->getError()
724
                ]);
725
            }
726
        } else {
727
            // Crear nueva conversación
728
            $conversation = new Conversation();
729
            $conversation->sender_id = $currentUser->id;
730
            $conversation->sender_status = Conversation::STATUS_NORMAL;
731
            $conversation->receiver_id = $user->id;
732
            $conversation->receiver_status = Conversation::STATUS_NORMAL;
582 ariadna 733
 
589 ariadna 734
            if (!$conversationMapper->insert($conversation)) {
735
                return new JsonModel([
736
                    'success' => false,
737
                    'data' => $conversationMapper->getError()
738
                ]);
739
            }
740
        }
307 www 741
 
589 ariadna 742
        // Procesar archivo adjunto si existe
743
        $files = $this->getRequest()->getFiles()->toArray();
744
        $type = Message::TYPE_TEXT;
745
        $message_tmp_filename = '';
746
        $message_filename = '';
307 www 747
 
589 ariadna 748
        if (isset($files['file']) && empty($files['file']['error'])) {
749
            $message_tmp_filename = $files['file']['tmp_name'];
750
            $message_filename = \LeadersLinked\Library\Functions::normalizeStringFilename($files['file']['name']);
307 www 751
 
589 ariadna 752
            // Determinar el tipo de archivo
753
            $mime_type = mime_content_type($message_tmp_filename);
754
            if ($mime_type == 'image/jpg' || $mime_type == 'image/jpeg' || $mime_type == 'image/png') {
755
                $type = Storage::FILE_TYPE_IMAGE;
756
            } else if ($mime_type == 'video/webm' || $mime_type == 'video/mpeg' || $mime_type == 'video/mpg' || $mime_type == 'video/mp4') {
757
                $type = Storage::FILE_TYPE_VIDEO;
758
            } else if ($mime_type == 'application/pdf') {
759
                $type = Storage::FILE_TYPE_DOCUMENT;
760
            }
761
        }
307 www 762
 
589 ariadna 763
        // Crear y guardar el mensaje
764
        $message = new Message();
765
        $message->conversation_id = $conversation->id;
766
        $message->read = Message::NO;
767
        $message->message = $type == Message::TYPE_TEXT ? $dataPost['message'] : '';
768
        $message->receiver_id = $user->id;
769
        $message->receiver_status = Message::STATUS_NORMAL;
770
        $message->sender_id = $currentUser->id;
771
        $message->sender_status = Message::STATUS_NORMAL;
772
        $message->type = $type;
307 www 773
 
589 ariadna 774
        $messageMapper = MessageMapper::getInstance($this->adapter);
775
        if (!$messageMapper->insert($message)) {
776
            return new JsonModel([
777
                'success' => false,
778
                'data' => $messageMapper->getError()
779
            ]);
780
        }
307 www 781
 
589 ariadna 782
        // Procesar archivo adjunto si existe
783
        if ($message_filename) {
784
            $storage = new Storage();
785
            $storage->setAdapter($this->adapter);
786
            $storage->setLogger($this->logger);
787
            $storage->setTranslator($this->translator);
307 www 788
 
589 ariadna 789
            if (!$storage->uploadMessageFile($message_tmp_filename, $message_filename, $message->uuid)) {
790
                return new JsonModel([
791
                    'success' => false,
792
                    'data' => $storage->getError()
793
                ]);
794
            }
307 www 795
 
589 ariadna 796
            $message->filename = $message_filename;
797
            if (!$messageMapper->update($message)) {
1 efrain 798
                return new JsonModel([
589 ariadna 799
                    'success' => false,
800
                    'data' => $messageMapper->getError()
1 efrain 801
                ]);
802
            }
803
        }
589 ariadna 804
 
805
        // Preparar respuesta exitosa
806
        $filename = '';
807
        if ($message->filename) {
808
            $filename = $this->url()->fromRoute('storage', [
809
                'type' => 'message',
810
                'filename' => $message->filename,
811
                'code' => $message->uuid
812
            ], ['force_canonical' => true]);
813
        }
814
 
815
        // Actualizar última actividad del usuario
816
        $userMapper->updateLastActivity($currentUser->id);
817
 
818
        // Devolver respuesta exitosa con datos del mensaje
819
        return new JsonModel([
820
            'success' => true,
821
            'data' => [
822
                'sender_name' => trim($currentUser->first_name . ' ' . $currentUser->last_name),
823
                'sender_image' => $this->url()->fromRoute('storage', [
824
                    'type' => 'user',
825
                    'filename' => $currentUser->image,
826
                    'code' => $currentUser->uuid
827
                ], ['force_canonical' => true]),
828
                'sender_profile' => $this->url()->fromRoute('profile/view', ['id' => $currentUser->uuid]),
829
                'receiver_name' => trim($user->first_name . ' ' . $user->last_name),
830
                'receiver_image' => $this->url()->fromRoute('storage', [
831
                    'type' => 'user',
832
                    'filename' => $user->image,
833
                    'code' => $user->uuid
834
                ], ['force_canonical' => true]),
835
                'receiver_profile' => $this->url()->fromRoute('profile/view', ['id' => $user->uuid]),
836
                'side' => 'left',
837
                'message' => $message->message,
838
                'type' => $message->type,
839
                'filename' => $filename,
840
                'date' => $this->timeAgo($now, $now),
841
            ]
842
        ]);
1 efrain 843
    }
307 www 844
 
845
    /**
846
     *
847
     * @param string $timestamp
848
     * @param string $now
849
     * @return string
850
     */
851
    private function timeAgo($timestamp, $now = '')
852
    {
582 ariadna 853
 
307 www 854
        if ($now) {
855
            $datetime1 = \DateTime::createFromFormat('Y-m-d H:i:s', $now);
1 efrain 856
        } else {
307 www 857
            $now = date('Y-m-d H:i:s');
858
            $datetime1 = date_create($now);
1 efrain 859
        }
307 www 860
        $datetime2 = date_create($timestamp);
582 ariadna 861
 
307 www 862
        $diff = date_diff($datetime1, $datetime2);
863
        $timemsg = '';
864
        if ($diff->y > 0) {
582 ariadna 865
            $timemsg = $diff->y . ' ' . ($diff->y > 1 ? $this->translator->translate('LABEL_YEARS_SMALL')  : $this->translator->translate('LABEL_YEAR_SMALL'));
307 www 866
        } else if ($diff->m > 0) {
582 ariadna 867
            $timemsg = $diff->m  . ' ' . ($diff->m > 1 ? $this->translator->translate('LABEL_MONTHS_SMALL')  : $this->translator->translate('LABEL_MONTH_SMALL'));
307 www 868
        } else if ($diff->d > 0) {
582 ariadna 869
            $timemsg = $diff->d . ' ' . ($diff->d > 1 ? $this->translator->translate('LABEL_DAYS_SMALL')  : $this->translator->translate('LABEL_DAY_SMALL'));
307 www 870
        } else if ($diff->h > 0) {
582 ariadna 871
            $timemsg = $diff->h  . ' ' . ($diff->h > 1 ? $this->translator->translate('LABEL_HOURS_SMALL')  : $this->translator->translate('LABEL_HOUR_SMALL'));
307 www 872
        } else if ($diff->i > 0) {
582 ariadna 873
            $timemsg = $diff->i  . ' ' . ($diff->i > 1 ? $this->translator->translate('LABEL_MINUTES_SMALL')  : $this->translator->translate('LABEL_MINUTE_SMALL'));
307 www 874
        } else if ($diff->s > 0) {
582 ariadna 875
            $timemsg = $diff->s  . ' ' . ($diff->s > 1 ? $this->translator->translate('LABEL_SECONDS_SMALL')  : $this->translator->translate('LABEL_SECOND_SMALL'));
307 www 876
        }
877
        if (!$timemsg) {
582 ariadna 878
            $timemsg = $this->translator->translate('LABEL_NOW');
307 www 879
        } else {
582 ariadna 880
            $timemsg = $this->translator->translate('LABEL_AGO_SMALL') . ' ' . $timemsg . '';
307 www 881
        }
882
        return $timemsg;
1 efrain 883
    }
307 www 884
 
885
    public function deleteAction()
1 efrain 886
    {
887
        $request = $this->getRequest();
888
        if ($request->isPost()) {
889
            $currentUserPlugin = $this->plugin('currentUserPlugin');
890
            $currentUser = $currentUserPlugin->getUser();
582 ariadna 891
 
1 efrain 892
            $id = $this->params()->fromRoute('id');
893
            if (!$id) {
894
                return new JsonModel([
895
                    'success' => false,
896
                    'data' => 'ERROR_PARAMETERS_ARE_INVALID'
897
                ]);
898
            }
582 ariadna 899
 
307 www 900
            $userMapper = UserMapper::getInstance($this->adapter);
901
            $user = $userMapper->fetchOneByUuid($id);
1 efrain 902
            if (!$user) {
903
                return new JsonModel([
904
                    'success' => false,
905
                    'data' => 'ERROR_REQUEST_IS_INVALID'
906
                ]);
907
            }
582 ariadna 908
 
307 www 909
            $conversationMapper = ConversationMapper::getInstance($this->adapter);
910
            $conversation = $conversationMapper->fetchOneByUserId1AndUserId2($currentUser->id, $user->id);
582 ariadna 911
 
912
 
307 www 913
            if ($conversation) {
582 ariadna 914
 
307 www 915
                if ($conversation->sender_id == $currentUser->id && $conversation->receiver_id == $user->id) {
916
                    $conversation->sender_status = Conversation::STATUS_DELETED;
917
                    if ($conversationMapper->update($conversation)) {
918
                        $response = [
919
                            'success' => true,
920
                            'data' => 'LABEL_CONVERSATION_WAS_DELETED'
921
                        ];
922
                    } else {
582 ariadna 923
 
924
 
307 www 925
                        $response = [
926
                            'success' => false,
927
                            'data' => $conversationMapper->getError()
928
                        ];
291 www 929
                    }
307 www 930
                }
582 ariadna 931
 
307 www 932
                if ($conversation->receiver_id == $currentUser->id && $conversation->sender_id == $user->id) {
933
                    $conversation->receiver_status = Conversation::STATUS_DELETED;
934
                    if ($conversationMapper->update($conversation)) {
935
                        $response = [
936
                            'success' => true,
937
                            'data' => 'LABEL_CONVERSATION_WAS_DELETED'
938
                        ];
291 www 939
                    } else {
582 ariadna 940
 
941
 
307 www 942
                        $response = [
943
                            'success' => false,
944
                            'data' => $conversationMapper->getError()
945
                        ];
1 efrain 946
                    }
947
                }
582 ariadna 948
 
307 www 949
                return new JsonModel($response);
1 efrain 950
            } else {
307 www 951
                $response = [
952
                    'success' => false,
953
                    'data' => 'ERROR_CONVERSATION_NOT_FOUND'
954
                ];
1 efrain 955
            }
582 ariadna 956
 
307 www 957
            return new JsonModel($response);
1 efrain 958
        } else {
307 www 959
            $response = [
1 efrain 960
                'success' => false,
961
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
307 www 962
            ];
1 efrain 963
        }
582 ariadna 964
 
307 www 965
        return new JsonModel($response);
1 efrain 966
    }
967
}