Proyectos de Subversion LeadersLinked - Backend

Rev

Rev 17050 | Autoría | Comparar con el anterior | Ultima modificación | Ver Log |

<?php
declare(strict_types=1);

namespace LeadersLinked\Controller;

use Laminas\Db\Adapter\AdapterInterface;

use Laminas\Mvc\Controller\AbstractActionController;
use Laminas\Log\LoggerInterface;
use Laminas\View\Model\ViewModel;
use Laminas\View\Model\JsonModel;
use LeadersLinked\Library\Functions;
use LeadersLinked\Hydrator\ObjectPropertyHydrator;
use LeadersLinked\Mapper\DailyPulseEmojiMapper;
use LeadersLinked\Form\DailyPulse\DailyPulseAddEmojiForm;
use LeadersLinked\Form\DailyPulse\DailyPulseEditEmojiForm;
use LeadersLinked\Model\DailyPulseEmoji;
use LeadersLinked\Library\Image;
use LeadersLinked\Library\Storage;

class DailyPulseEmojiController extends AbstractActionController
{
    /**
     *
     * @var \Laminas\Db\Adapter\AdapterInterface
     */
    private $adapter;
    
    /**
     *
     * @var \LeadersLinked\Cache\CacheInterface
     */
    private $cache;
    
    
    /**
     *
     * @var \Laminas\Log\LoggerInterface
     */
    private $logger;
    
    /**
     *
     * @var array
     */
    private $config;
    
    
    /**
     *
     * @var \Laminas\Mvc\I18n\Translator
     */
    private $translator;
    
    
    /**
     *
     * @param \Laminas\Db\Adapter\AdapterInterface $adapter
     * @param \LeadersLinked\Cache\CacheInterface $cache
     * @param \Laminas\Log\LoggerInterface LoggerInterface $logger
     * @param array $config
     * @param \Laminas\Mvc\I18n\Translator $translator
     */
    public function __construct($adapter, $cache, $logger, $config, $translator)
    {
        $this->adapter      = $adapter;
        $this->cache        = $cache;
        $this->logger       = $logger;
        $this->config       = $config;
        $this->translator   = $translator;
    }
    
    public function indexAction()
    {
        $currentNetworkPlugin = $this->plugin('currentNetworkPlugin');
        $currentNetwork = $currentNetworkPlugin->getNetwork();
        
        $currentUserPlugin  = $this->plugin('currentUserPlugin');
        $currentUser        = $currentUserPlugin->getUser();
        $currentCompany     = $currentUserPlugin->getCompany();
        
        $request = $this->getRequest();
        if($request->isGet()) {
            
            
            $headers  = $request->getHeaders();
            
            $isJson = false;
            if($headers->has('Accept')) {
                $accept = $headers->get('Accept');
                
                $prioritized = $accept->getPrioritized();
                
                foreach($prioritized as $key => $value) {
                    $raw = trim($value->getRaw());
                    
                    if(!$isJson) {
                        $isJson = strpos($raw, 'json');
                    }
                    
                }
            }
            
            if($isJson) {
                $search = $this->params()->fromQuery('search', []);
                $search = empty($search['value']) ? '' :  Functions::sanitizeFilterString($search['value']);
                
                $page               = intval($this->params()->fromQuery('start', 1), 10);
                $records_x_page     = intval($this->params()->fromQuery('length', 10), 10);
                $order =  $this->params()->fromQuery('order', []);
                $order_field        = empty($order[0]['column']) ? 99 :  intval($order[0]['column'], 10);
                $order_direction    = empty($order[0]['dir']) ? 'ASC' : strtoupper(Functions::sanitizeFilterString($order[0]['dir']));
                
                $fields =  ['name'];
                $order_field = isset($fields[$order_field]) ? $fields[$order_field] : 'name';
                
                if(!in_array($order_direction, ['ASC', 'DESC'])) {
                    $order_direction = 'ASC';
                }
                
                $dailyPulseEmojiMapper = DailyPulseEmojiMapper::getInstance($this->adapter);
                $paginator = $dailyPulseEmojiMapper->fetchAllDataTable($currentCompany->id, $search,  $page, $records_x_page, $order_field, $order_direction);

                $storage = Storage::getInstance($this->config, $this->adapter);
                $path = $storage->getPathDailyPulse();
                
                $items = [];
                $records = $paginator->getCurrentItems();
                foreach($records as $record)
                {
                    switch($record->type)
                    {
                        case DailyPulseEmoji::TYPE_HOW_ARE_YOU_FEEL : 
                            $type = 'LABEL_HOW_ARE_YOU_FEEL';
                            break;
                     
                        case DailyPulseEmoji::TYPE_CLIMATE_ON_YOUR_ORGANIZATION :
                            $type = 'LABEL_CLIMATE_ON_YOUR_ORGANIZATION';
                            break;
                            
                        default : 
                            $type = 'LABEL_UNKNOWN';
                            break;
                            
                    }
                    
                    switch($record->status) 
                    {
                        case DailyPulseEmoji::STATUS_ACTIVE :
                            $status = 'LABEL_ACTIVE';
                            break;
                            
                        case DailyPulseEmoji::STATUS_INACTIVE :
                            $status = 'LABEL_INACTIVE';
                            break;
                            
                        default :
                            $status = 'LABEL_UNKNOWN';
                            break;
                    }
                    
                    
                    
                    $item = [
                        'name' => $record->name,
                        'details' => [
                            'type' => $type,
                            'status' => $status,
                            'order' => $record->order,
                            'points' => $record->points,
                        ],    
                        'image' => $storage->getGenericImage($path, $record->uuid, $record->image),
                        'actions' => [
                            'link_edit' => $this->url()->fromRoute('daily-pulse/emojis/edit', ['id' => $record->uuid ]),
                            'link_delete' => $this->url()->fromRoute('daily-pulse/emojis/delete', ['id' => $record->uuid ]),
                        ]
                    ];
                    
                    array_push($items, $item);
                }
                
                return new JsonModel([
                    'success' => true,
                    'data' => [
                        'items' => $items,
                        'total' => $paginator->getTotalItemCount(),
                    ]
                ]);
            } else  {
                
                $target_size = $this->config['leaderslinked.image_sizes.emoji'];
                
                $formAdd = new DailyPulseAddEmojiForm();
                $formEdit = new DailyPulseEditEmojiForm();
                
                $this->layout()->setTemplate('layout/layout-backend');
                $viewModel = new ViewModel();
                $viewModel->setTemplate('leaders-linked/daily-pulse/emojis');
                $viewModel->setVariables([
                    'formAdd' => $formAdd,
                    'formEdit' => $formEdit,
                    'targetSize' => $target_size,
                ]);
                return $viewModel ;
            }
        } else {
            return new JsonModel([
                'success' => false,
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
            ]);;
        }
    }
    
    
    
    /**
     * Agregar un nuevo emoji para el pulso diario
     * 
     * Este método maneja la creación de un nuevo registro de emoji con su imagen asociada.
     * Valida los datos del formulario, crea el registro del emoji y maneja la carga de la imagen.
     * 
     * @return JsonModel Respuesta con estado de éxito/error y mensaje
     */
    public function addAction()
    {
        try {
            // Validar método de solicitud
            if (!$this->getRequest()->isPost()) {
                return $this->createErrorResponse('ERROR_METHOD_NOT_ALLOWED');
            }

            // Obtener y validar datos del formulario
            $formData = $this->getFormData();
            if (!$formData['isValid']) {
                return $this->createErrorResponse($formData['messages']);
            }

            // Crear y guardar el emoji
            $emoji = $this->createEmoji($formData['data']);
            if (!$emoji) {
                return $this->createErrorResponse('ERROR_CREATING_EMOJI');
            }

            // Procesar y guardar la imagen
            if (!$this->processAndSaveImage($emoji)) {
                return $this->createErrorResponse('ERROR_UPLOAD_FILE');
            }

            // Registrar la acción y retornar éxito
            $this->logEmojiCreation($emoji);
            return $this->createSuccessResponse('LABEL_RECORD_ADDED');

        } catch (\Exception $e) {
            $this->logger->error('Error al crear emoji: ' . $e->getMessage());
            return $this->createErrorResponse('ERROR_UNEXPECTED');
        }
    }
    
    /**
     * Obtiene y valida los datos del formulario
     * 
     * @return array Datos validados y estado de validación
     */
    private function getFormData()
    {
        $form = new DailyPulseAddEmojiForm();
        $dataPost = array_merge(
            $this->getRequest()->getPost()->toArray(),
            $this->getRequest()->getFiles()->toArray()
        );
        
        $dataPost['status'] = empty($dataPost['status']) 
            ? DailyPulseEmoji::STATUS_INACTIVE 
            : $dataPost['status'];
        
        $form->setData($dataPost);
        
        if ($form->isValid()) {
            return [
                'isValid' => true,
                'data' => (array) $form->getData()
            ];
        }

        $messages = [];
        foreach ((array) $form->getMessages() as $fieldname => $field_messages) {
            $messages[$fieldname] = array_values($field_messages);
        }

        return [
            'isValid' => false,
            'messages' => $messages
        ];
    }

    /**
     * Crea un nuevo registro de emoji
     * 
     * @param array $data Datos del emoji
     * @return DailyPulseEmoji|null Emoji creado o null si falla
     */
    private function createEmoji(array $data)
    {
        $currentCompany = $this->plugin('currentUserPlugin')->getCompany();
        
        $emoji = new DailyPulseEmoji();
        $emoji->company_id = $currentCompany->id;
        $emoji->name = $data['name'];
        $emoji->order = $data['order'];
        $emoji->points = $data['points'];
        $emoji->status = $data['status'];
        $emoji->type = $data['type'];
        $emoji->image = '';

        $mapper = DailyPulseEmojiMapper::getInstance($this->adapter);
        if (!$mapper->insert($emoji)) {
            return null;
        }

        return $mapper->fetchOne($emoji->id);
    }

    /**
     * Procesa y guarda la imagen del emoji
     * 
     * @param DailyPulseEmoji $emoji Emoji al que se le asignará la imagen
     * @return bool True si la imagen se procesó correctamente
     */
    private function processAndSaveImage(DailyPulseEmoji $emoji)
    {
        $storage = Storage::getInstance($this->config, $this->adapter);
        $storage->setFiles($this->getRequest()->getFiles()->toArray());

        if (!$storage->setCurrentFilename('image')) {
            return false;
        }

        $target_size = $this->config['leaderslinked.image_sizes.emoji'];
        list($target_width, $target_height) = explode('x', $target_size);

        $source_filename = $storage->getTmpFilename();
        $filename = 'emoji-' . uniqid() . '.png';
        $target_filename = $storage->composePathToFilename(
            Storage::TYPE_DAILY_PULSE,
            $emoji->uuid,
            $filename
        );

        if (!$storage->uploadImageCrop($source_filename, $target_filename, $target_width, $target_height)) {
            return false;
        }

        $emoji->image = $filename;
        return DailyPulseEmojiMapper::getInstance($this->adapter)->update($emoji);
    }

    /**
     * Registra la creación del emoji en el log
     * 
     * @param DailyPulseEmoji $emoji Emoji creado
     */
    private function logEmojiCreation(DailyPulseEmoji $emoji)
    {
        $currentUser = $this->plugin('currentUserPlugin')->getUser();
        $this->logger->info(
            'Se agrego el emoji ' . $emoji->name,
            [
                'user_id' => $currentUser->id,
                'ip' => Functions::getUserIP()
            ]
        );
    }

    /**
     * Crea una respuesta de error
     * 
     * @param mixed $message Mensaje de error
     * @return JsonModel
     */
    private function createErrorResponse($message)
    {
        return new JsonModel([
            'success' => false,
            'data' => $message
        ]);
    }

    /**
     * Crea una respuesta de éxito
     * 
     * @param string $message Mensaje de éxito
     * @return JsonModel
     */
    private function createSuccessResponse($message)
    {
        return new JsonModel([
            'success' => true,
            'data' => $message
        ]);
    }
    
    /**
     *
     * Borrar un perfil excepto el público
     * @return \Laminas\View\Model\JsonModel
     */
    public function deleteAction()
    {
        $currentUserPlugin = $this->plugin('currentUserPlugin');
        $currentUser    = $currentUserPlugin->getUser();
        $currentCompany = $currentUserPlugin->getCompany();
        
        $request        = $this->getRequest();
        $id         = $this->params()->fromRoute('id');

        

        $dailyPulseEmojiMapper = DailyPulseEmojiMapper::getInstance($this->adapter);
        $dailyPulseEmoji = $dailyPulseEmojiMapper->fetchOneByUuid($id);
        if(!$dailyPulseEmoji) {
            return new JsonModel([
                'success'   => false,
                'data'   => 'ERROR_RECORD_NOT_FOUND'
            ]);
        }
        
        if($dailyPulseEmoji->company_id != $currentCompany->id) {
            return new JsonModel([
                'success'   => false,
                'data'   => 'ERROR_UNAUTHORIZED'
            ]);
        }
        
        if($request->isPost()) {
            
            $result =  $dailyPulseEmojiMapper->delete($dailyPulseEmoji);
            if($result) {
                $this->logger->info('Se borro el emoji : ' .  $dailyPulseEmoji->name, ['user_id' => $currentUser->id, 'ip' => Functions::getUserIP()]);
               
                if($dailyPulseEmoji->image) {
                
                    $storage = Storage::getInstance($this->config, $this->adapter);
                    $target_path = $storage->getPathDailyPulse();
                    $storage->deleteFile($target_path, $dailyPulseEmoji->uuid, $dailyPulseEmoji->image);
                }
                
                $data = [
                    'success' => true,
                    'data' => 'LABEL_RECORD_DELETED'
                ];
            } else {
                
                $data = [
                    'success'   => false,
                    'data'      => $dailyPulseEmojiMapper->getError()
                ];
                
                return new JsonModel($data);
            }
            
        } else {
            $data = [
                'success' => false,
                'data' => 'ERROR_METHOD_NOT_ALLOWED'
            ];
            
            return new JsonModel($data);
        }
        
        return new JsonModel($data);
    }
    
    
    /**
     * Editar un emoji existente
     * 
     * Este método maneja la actualización de un emoji existente y su imagen asociada.
     * Valida los datos del formulario, actualiza el registro del emoji y maneja la carga de la imagen.
     * 
     * @return JsonModel Respuesta con estado de éxito/error y mensaje
     */
    public function editAction()
    {
        try {
            $id = $this->params()->fromRoute('id');
            
            // Obtener y validar el emoji existente
            $emoji = $this->getExistingEmoji($id);
            if (!$emoji) {
                return $this->createErrorResponse('ERROR_RECORD_NOT_FOUND');
            }

            // Validar autorización
            if (!$this->isAuthorized($emoji)) {
                return $this->createErrorResponse('ERROR_UNAUTHORIZED');
            }

            $request = $this->getRequest();
            
            // Manejar solicitud GET
            if ($request->isGet()) {
                return $this->getEmojiData($emoji);
            }
            
            // Manejar solicitud POST
            if ($request->isPost()) {
                return $this->updateEmoji($emoji);
            }

            return $this->createErrorResponse('ERROR_METHOD_NOT_ALLOWED');

        } catch (\Exception $e) {
            $this->logger->error('Error al editar emoji: ' . $e->getMessage());
            return $this->createErrorResponse('ERROR_UNEXPECTED');
        }
    }

    /**
     * Obtiene un emoji existente por su UUID
     * 
     * @param string $id UUID del emoji
     * @return DailyPulseEmoji|null Emoji encontrado o null
     */
    private function getExistingEmoji($id)
    {
        $mapper = DailyPulseEmojiMapper::getInstance($this->adapter);
        return $mapper->fetchOneByUuid($id);
    }

    /**
     * Verifica si el usuario está autorizado para editar el emoji
     * 
     * @param DailyPulseEmoji $emoji Emoji a verificar
     * @return bool True si está autorizado
     */
    private function isAuthorized(DailyPulseEmoji $emoji)
    {
        $currentCompany = $this->plugin('currentUserPlugin')->getCompany();
        return $emoji->company_id === $currentCompany->id;
    }

    /**
     * Obtiene los datos del emoji para la vista
     * 
     * @param DailyPulseEmoji $emoji Emoji a obtener datos
     * @return JsonModel Datos del emoji
     */
    private function getEmojiData(DailyPulseEmoji $emoji)
    {
        $storage = Storage::getInstance($this->config, $this->adapter);
        $path = $storage->getPathDailyPulse();

        return $this->createSuccessResponse([
            'name' => $emoji->name,
            'order' => $emoji->order,
            'status' => $emoji->status,
            'type' => $emoji->type,
            'points' => $emoji->points,
            'image' => $emoji->image ? $storage->getGenericImage($path, $emoji->uuid, $emoji->image) : ''
        ]);
    }

    /**
     * Actualiza un emoji existente
     * 
     * @param DailyPulseEmoji $emoji Emoji a actualizar
     * @return JsonModel Resultado de la actualización
     */
    private function updateEmoji(DailyPulseEmoji $emoji)
    {
        // Obtener y validar datos del formulario
        $formData = $this->getEditFormData();
        if (!$formData['isValid']) {
            return $this->createErrorResponse($formData['messages']);
        }

        // Actualizar datos del emoji
        $this->updateEmojiData($emoji, $formData['data']);

        // Procesar imagen si se proporcionó una nueva
        if ($this->hasNewImage()) {
            if (!$this->processAndUpdateImage($emoji)) {
                return $this->createErrorResponse('ERROR_UPLOAD_FILE');
            }
        }

        // Guardar cambios
        if (!$this->saveEmoji($emoji)) {
            return $this->createErrorResponse('ERROR_UPDATING_EMOJI');
        }

        // Registrar la acción
        $this->logEmojiUpdate($emoji);
        
        return $this->createSuccessResponse('LABEL_RECORD_UPDATED');
    }

    /**
     * Obtiene y valida los datos del formulario de edición
     * 
     * @return array Datos validados y estado de validación
     */
    private function getEditFormData()
    {
        $form = new DailyPulseEditEmojiForm();
        $dataPost = array_merge(
            $this->getRequest()->getPost()->toArray(),
            $this->getRequest()->getFiles()->toArray()
        );
        
        $dataPost['status'] = empty($dataPost['status']) 
            ? DailyPulseEmoji::STATUS_INACTIVE 
            : $dataPost['status'];
        
        $form->setData($dataPost);
        
        if ($form->isValid()) {
            return [
                'isValid' => true,
                'data' => (array) $form->getData()
            ];
        }

        $messages = [];
        foreach ((array) $form->getMessages() as $fieldname => $field_messages) {
            $messages[$fieldname] = array_values($field_messages);
        }

        return [
            'isValid' => false,
            'messages' => $messages
        ];
    }

    /**
     * Actualiza los datos del emoji con la información del formulario
     * 
     * @param DailyPulseEmoji $emoji Emoji a actualizar
     * @param array $data Datos del formulario
     */
    private function updateEmojiData(DailyPulseEmoji $emoji, array $data)
    {
        $emoji->name = $data['name'];
        $emoji->order = $data['order'];
        $emoji->points = $data['points'];
        $emoji->status = $data['status'];
        $emoji->type = $data['type'];
    }

    /**
     * Verifica si se proporcionó una nueva imagen
     * 
     * @return bool True si hay una nueva imagen
     */
    private function hasNewImage()
    {
        $files = $this->getRequest()->getFiles()->toArray();
        return isset($files['image']) && empty($files['image']['error']);
    }

    /**
     * Procesa y actualiza la imagen del emoji
     * 
     * @param DailyPulseEmoji $emoji Emoji a actualizar
     * @return bool True si la imagen se procesó correctamente
     */
    private function processAndUpdateImage(DailyPulseEmoji $emoji)
    {
        $storage = Storage::getInstance($this->config, $this->adapter);
        $target_path = $storage->getPathDailyPulse();
        
        if(!$storage->setCurrentFilename('image')) {
            return false;
        }
        
        $source_filename = $storage->getTmpFilename();
        $filename = 'emoji-' . uniqid() . '.png';
        $target_filename = $storage->composePathToFilename(
            Storage::TYPE_DAILY_PULSE,
            $emoji->uuid,
            $filename
        );
        
        $target_size = $this->config['leaderslinked.image_sizes.emoji'];
        list($target_width, $target_height) = explode('x', $target_size);
        
        $full_tmp_filename = $storage->uploadImageCrop(
            $source_filename,
            $target_filename,
            $target_width,
            $target_height
        );
        
        if (!$full_tmp_filename) {
            return false;
        }

        // Eliminar imagen anterior si existe
        if ($emoji->image) {
            $storage->deleteFile($target_path, $emoji->uuid, $emoji->image);
        }

        $emoji->image = $filename;
        return true;
    }

    /**
     * Guarda los cambios del emoji en la base de datos
     * 
     * @param DailyPulseEmoji $emoji Emoji a guardar
     * @return bool True si se guardó correctamente
     */
    private function saveEmoji(DailyPulseEmoji $emoji)
    {
        $mapper = DailyPulseEmojiMapper::getInstance($this->adapter);
        return $mapper->update($emoji);
    }

    /**
     * Registra la actualización del emoji en el log
     * 
     * @param DailyPulseEmoji $emoji Emoji actualizado
     */
    private function logEmojiUpdate(DailyPulseEmoji $emoji)
    {
        $currentUser = $this->plugin('currentUserPlugin')->getUser();
        $this->logger->info(
            'Se edito el emoji ' . $emoji->name,
            [
                'user_id' => $currentUser->id,
                'ip' => Functions::getUserIP()
            ]
        );
    }
}