<?php

namespace Leantime\Domain\Projects\Repositories;

use DateInterval;
use DatePeriod;
use Illuminate\Contracts\Container\BindingResolutionException;
use Leantime\Core\Configuration\Environment;
use Leantime\Core\Db\Db as DbCore;
use Leantime\Core\Events\DispatchesEvents as EventhelperCore;
use Leantime\Core\Support\Avatarcreator;
use Leantime\Domain\Auth\Models\Roles;
use Leantime\Domain\Users\Repositories\Users as UserRepository;
use PDO;
use SVG\SVG;

class Projects
{
    use EventhelperCore;

    public string $name = '';

    public int $id = 0; // WAS: '';

    public int $clientId = 0;

    public object $result; // WAS: = '';

    /**
     * @var array state for projects
     */
    public array $state = [0 => 'OPEN', 1 => 'CLOSED', null => 'OPEN'];

    public function __construct(
        protected Environment $config,
        protected DbCore $db,
        protected Avatarcreator $avatarcreator
    ) {
        $this->config = $config;
        $this->db = $db;
    }

    /**
     * getAll - get all projects open and closed
     */
    public function getAll(bool $showClosedProjects = false): array
    {

        $query = "SELECT
					project.id,
					project.name,
					project.clientId,
					project.hourBudget,
					project.dollarBudget,
					project.state,
                    project.menuType,
                    project.type,
                    project.modified,
					client.name AS clientName,
					client.id AS clientId,
					comments.status as status,
					project.start,
					project.end
				FROM zp_projects as project
				LEFT JOIN zp_clients as client ON project.clientId = client.id
                LEFT JOIN zp_comment as comments ON comments.id = (
                      SELECT
						id
                      FROM zp_comment
                      WHERE module = 'project' AND moduleId = project.id
                      ORDER BY date DESC LIMIT 1
                    )
				";

        if ($showClosedProjects === false) {
            $query .= ' WHERE project.state IS NULL OR project.state <> -1 ';
        }

        $query .= '
				GROUP BY
					project.id,
					project.name,
					project.clientId
				ORDER BY clientName, project.name';

        $stmn = $this->db->database->prepare($query);

        $stmn->execute();
        $values = $stmn->fetchAll();
        $stmn->closeCursor();

        return $values;
    }

    /**
     * Gets all users that have access to a project.
     * For direct access only set the teamOnly flag to true
     */
    public function getUsersAssignedToProject($id, $includeApiUsers = false): array|bool
    {

        $query = 'SELECT
					DISTINCT zp_user.id,
					IF(zp_user.firstname IS NOT NULL, zp_user.firstname, zp_user.username) AS firstname,
					zp_user.lastname,
					zp_user.username,
					zp_user.notifications,
					zp_user.profileId,
					zp_user.jobTitle,
					zp_user.source,
                    zp_user.status,
                    zp_user.modified,
                    zp_user.role,
                    zp_relationuserproject.projectRole
				FROM zp_relationuserproject
				LEFT JOIN zp_user ON zp_relationuserproject.userId = zp_user.id
				LEFT JOIN zp_projects ON zp_relationuserproject.projectId = zp_projects.id
                WHERE
                        zp_relationuserproject.projectId = :projectId
                        AND zp_user.id IS NOT NULL ';

        if ($includeApiUsers === false) {
            $query .= " AND !(zp_user.source <=> 'api') ";
        }

        $query .= '
				    GROUP BY zp_user.id
                    ORDER BY zp_user.lastname';

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':projectId', $id, PDO::PARAM_INT);

        $stmn->execute();
        $values = $stmn->fetchAll();
        $stmn->closeCursor();

        return $values;
    }

    /**
     * Retrieves the relationship of users assigned to a specific project.
     *
     * @param  int  $id  The ID of the project.
     * @param  bool  $includeApiUsers  Flag to determine whether to include API users. Default is false.
     * @return array|bool Returns an array of users assigned to the project or false on failure.
     *
     * @Deprecated
     */
    public function getProjectUserRelation($id, $includeApiUsers = false): array|bool
    {
        return $this->getUsersAssignedToProject($id, $includeApiUsers);
    }

    public function getUserProjects(int $userId, string $projectStatus = 'all', ?int $clientId = null, string $accessStatus = 'assigned', string $projectTypes = 'all'): false|array
    {

        $query = "SELECT
					project.id,
					project.name,
					project.details,
					project.clientId,
					project.state,
					project.hourBudget,
					project.dollarBudget,
				    project.menuType,
				    project.type,
				    project.parent,
				    project.modified,
				    project.start,
				    project.end,
					client.name AS clientName,
					client.id AS clientId,
					parent.id AS parentId,
					parent.name as parentName,
					IF(favorite.id IS NULL, false, true) as isFavorite
				FROM zp_projects as project
				LEFT JOIN zp_relationuserproject AS relation ON project.id = relation.projectId
				LEFT JOIN zp_projects as parent ON parent.id = project.parent
				LEFT JOIN zp_clients as client ON project.clientId = client.id
				LEFT JOIN zp_user as `user` ON relation.userId = user.id
				LEFT JOIN zp_reactions as favorite ON project.id = favorite.moduleId
				                                          AND favorite.module = 'project'
				                                          AND favorite.reaction = 'favorite'
				                                          AND favorite.userId = :id
                LEFT JOIN zp_user as requestingUser ON requestingUser.id = :id
				WHERE (project.active > '-1' OR project.active IS NULL)";

        // All Projects this user has access to
        if ($accessStatus == 'all') {
            $query .= " AND
				(
				    relation.userId = :id
                    OR (project.psettings = 'clients' AND project.clientId = requestingUser.clientId)
                    OR project.psettings = 'all'
                    OR requestingUser.role >= 40

				)";

            // All projects the user is assigned to OR the users client is assigned to
        } elseif ($accessStatus == 'clients') {
            $query .= " AND
				(
				    relation.userId = :id
                    OR (project.psettings = 'clients' AND project.clientId = requestingUser.clientId)
				)";

            // Only assigned
        } else {
            $query .= ' AND
				(relation.userId = :id)';
        }

        if ($projectStatus == 'open') {
            $query .= " AND (project.state <> '-1' OR project.state IS NULL)";
        } elseif ($projectStatus == 'closed') {
            $query .= ' AND (project.state = -1)';
        }

        if ($clientId != '' && $clientId != null && $clientId > 0) {
            $query .= ' AND project.clientId = :clientId';
        }

        if ($projectTypes != 'all' && $projectTypes != 'project') {
            $projectTypeIn = DbCore::arrayToPdoBindingString('projectType', count(explode(',', $projectTypes)));
            $query .= ' AND project.type IN('.$projectTypeIn.')';
        }

        if ($projectTypes == 'project') {
            $query .= " AND (project.type = 'project' OR project.type IS NULL)";
        }

        $query .= ' GROUP BY
					project.id
				    ORDER BY clientName, project.name';

        $stmn = $this->db->database->prepare($query);

        if ($userId == '') {
            $stmn->bindValue(':id', session('userdata.id'), PDO::PARAM_STR);
        } else {
            $stmn->bindValue(':id', $userId, PDO::PARAM_STR);
        }

        if ($clientId != '' && $clientId != null && $clientId > 0) {
            $stmn->bindValue(':clientId', $clientId, PDO::PARAM_STR);
        }

        if ($projectTypes != 'all' && $projectTypes != 'project') {
            foreach (explode(',', $projectTypes) as $key => $type) {
                $stmn->bindValue(':projectType'.$key, $type, PDO::PARAM_STR);
            }
        }

        if ($projectTypes == 'project') {
            $query .= " AND (project.type = 'project' OR project.type IS NULL)";
        }

        $stmn->execute();
        $values = $stmn->fetchAll(PDO::FETCH_ASSOC);
        $stmn->closeCursor();

        return $values;
    }

    // This populates the projects show all tab and shows users all the projects that they could access
    public function getProjectsUserHasAccessTo($userId, string $status = 'all', string $clientId = ''): false|array
    {

        $query = "SELECT
					project.id,
					project.name,
					project.clientId,
					project.state,
					project.hourBudget,
					project.dollarBudget,
				    project.menuType,
				    project.type,
				    project.parent,
				    project.modified,
					client.name AS clientName,
					client.id AS clientId,
					IF(favorite.id IS NULL, false, true) as isFavorite
				FROM zp_projects AS project
				LEFT JOIN zp_relationuserproject as relation ON project.id = relation.projectId
				LEFT JOIN zp_clients as client ON project.clientId = client.id
				LEFT JOIN zp_reactions as favorite ON project.id = favorite.moduleId AND favorite.module = 'project' AND favorite.reaction = 'favorite' AND favorite.userId = :id
				WHERE
				    (   relation.userId = :id
				        OR project.psettings = 'all'
				        OR (project.psettings = 'clients' AND project.clientId = :clientId)
				    )
				  AND (project.active > '-1' OR project.active IS NULL)";

        if ($status == 'open') {
            $query .= " AND (project.state <> '-1' OR project.state IS NULL)";
        } elseif ($status == 'closed') {
            $query .= ' AND (project.state = -1)';
        }

        $query .= ' GROUP BY
					project.id
				ORDER BY clientName, project.name';

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':id', $userId, PDO::PARAM_STR);
        $stmn->bindValue(':clientId', $clientId, PDO::PARAM_STR);

        $stmn->execute();
        $values = $stmn->fetchAll();
        $stmn->closeCursor();

        return $values;
    }

    /**
     * @return int|mixed
     */
    public function getNumberOfProjects($clientId = null, $type = null): mixed
    {

        $sql = 'SELECT COUNT(id) AS projectCount FROM `zp_projects` WHERE id >0';

        if ($clientId != null && is_numeric($clientId)) {
            $sql .= ' AND clientId = :clientId';
        }

        if ($type != null) {
            $sql .= ' AND type = :type';
        }

        $stmn = $this->db->database->prepare($sql);

        if ($clientId != null && is_numeric($clientId)) {
            $stmn->bindValue(':clientId', $clientId, PDO::PARAM_INT);
        }

        if ($type != null) {
            $stmn->bindValue(':type', $type, PDO::PARAM_STR);
        }

        $stmn->execute();
        $values = $stmn->fetch();
        $stmn->closeCursor();

        if (isset($values['projectCount']) === true) {
            return $values['projectCount'];
        } else {
            return 0;
        }
    }

    // Get all open user projects /param: open, closed, all

    public function getClientProjects($clientId): false|array
    {

        $sql = "SELECT
					project.id,
					project.name,
					project.clientId,
					project.hourBudget,
					project.dollarBudget,
					project.state,
				    project.menuType,
				    project.modified,
				    project.type,
					client.name AS clientName,
					client.id AS clientId
				FROM zp_projects as project
				LEFT JOIN zp_clients as client ON project.clientId = client.id
				WHERE
				  (project.active > '-1' OR project.active IS NULL)
				  AND clientId = :clientId
				GROUP BY
					project.id,
					project.name,
					project.clientId
				ORDER BY clientName, project.name";

        $stmn = $this->db->database->prepare($sql);
        $stmn->bindValue(':clientId', $clientId, PDO::PARAM_INT);

        $stmn->execute();
        $values = $stmn->fetchAll();
        $stmn->closeCursor();

        return $values;
    }

    public function getProjectTickets($projectId): false|array
    {

        $sql = 'SELECT zp_tickets.id,
		zp_tickets.headline,
		zp_tickets.editFrom,
		zp_tickets.editTo,
		zp_user.firstname,
		zp_user.lastname
		 FROM zp_tickets
		LEFT JOIN zp_user ON zp_tickets.editorId = zp_user.id
		WHERE projectId=:projectId ORDER BY zp_tickets.editFrom';

        $stmn = $this->db->database->prepare($sql);
        $stmn->bindValue(':projectId', $projectId, PDO::PARAM_INT);

        $stmn->execute();
        $values = $stmn->fetchAll();
        $stmn->closeCursor();

        return $values;
    }

    /**
     * getProject - get one project
     */
    public function getProject($id): array|bool
    {

        $query = "SELECT
					zp_projects.id,
					zp_projects.name,
					zp_projects.clientId,
					zp_projects.details,
					zp_projects.state,
					zp_projects.hourBudget,
					zp_projects.dollarBudget,
					zp_projects.psettings,
				    zp_projects.menuType,
				    zp_projects.avatar,
				    zp_projects.cover,
				    zp_projects.type,
				    zp_projects.parent,
				    zp_projects.modified,
					zp_clients.name AS clientName,
					zp_projects.start,
					zp_projects.end,
                    IF(favorite.id IS NULL, false, true) as isFavorite
				FROM zp_projects
				  LEFT JOIN zp_clients ON zp_projects.clientId = zp_clients.id
				  LEFT JOIN zp_reactions as favorite ON zp_projects.id = favorite.moduleId
				                                          AND favorite.module = 'project'
				                                          AND favorite.reaction = 'favorite'
				                                          AND favorite.userId = :id
                  LEFT JOIN zp_user as requestingUser ON requestingUser.id = :id
				WHERE zp_projects.id = :projectId
				GROUP BY
					zp_projects.id,
					zp_projects.name,
					zp_projects.clientId,
					zp_projects.details
				LIMIT 1";

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':projectId', $id, PDO::PARAM_INT);
        $stmn->bindValue(':id', session('userdata.id'), PDO::PARAM_STR);

        $stmn->execute();
        $values = $stmn->fetch();
        $stmn->closeCursor();

        return $values;
    }

    public function getProjectBookedHours($id): array|bool
    {

        $query = 'SELECT zp_tickets.projectId, SUM(zp_timesheets.hours) AS totalHours
				FROM zp_tickets
				INNER JOIN zp_timesheets ON zp_timesheets.ticketId = zp_tickets.id
				WHERE projectId = :id';

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':id', $id, PDO::PARAM_INT);

        $stmn->execute();
        $values = $stmn->fetch();
        $stmn->closeCursor();

        return $values;
    }

    public function recursive_array_search($needle, $haystack): false|int|string
    {
        foreach ($haystack as $key => $value) {
            $current_key = $key;
            if ($needle === $value or (is_array($value) && $this->recursive_array_search($needle, $value) !== false)) {
                return $current_key;
            }
        }

        return false;
    }

    public function getProjectBookedHoursArray($id): array|bool
    {

        $query = "SELECT
                        zp_tickets.projectId,
			            SUM(zp_timesheets.hours) AS totalHours,
			            DATE_FORMAT(zp_timesheets.workDate,'%Y-%m-%d') AS workDate
				    FROM zp_tickets
				    INNER JOIN zp_timesheets ON zp_timesheets.ticketId = zp_tickets.id
				    WHERE projectId =  :id GROUP BY zp_timesheets.workDate
				    ORDER BY workDate";

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':id', $id, PDO::PARAM_INT);

        $stmn->execute();
        $results = $stmn->fetchAll();
        $stmn->closeCursor();

        $chartArr = [];

        if (count($results) > 0) {
            $begin = date_create($results[0]['workDate']);
            $begin->sub(new DateInterval('P1D'));

            $end = date_create($results[(count($results) - 1)]['workDate']);
            $end->add(new DateInterval('P1D'));

            $i = new DateInterval('P1D');

            $period = new DatePeriod($begin, $i, $end);

            $total = 0;

            foreach ($period as $d) {
                $day = $d->format('Y-m-d');
                $dayKey = $d->getTimestamp();

                $key = $this->recursive_array_search($day, $results);

                if ($key === false) {
                    $value = 0;
                } else {
                    $value = $results[$key]['totalHours'];
                }

                $total = $total + $value;
                $chartArr[$dayKey] = $total;
            }
        }

        return $chartArr;
    }

    public function getProjectBookedDollars($id): mixed
    {

        $query = 'SELECT zp_tickets.projectId, SUM(zp_timesheets.hours*zp_timesheets.rate) AS totalDollars
				FROM zp_tickets
				INNER JOIN zp_timesheets ON zp_timesheets.ticketId = zp_tickets.id
				WHERE projectId = :id';

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':id', $id, PDO::PARAM_INT);

        $stmn->execute();
        $values = $stmn->fetch();
        $stmn->closeCursor();

        return $values;
    }

    /**
     * addProject - add a project to a client
     *
     * @return int|bool returns new project id on success, false on failure.
     */
    public function addProject(bool|array $values): int|bool
    {

        $query = 'INSERT INTO `zp_projects` (
				            `name`,
                           `details`,
                           `clientId`,
                           `hourBudget`,
                           `dollarBudget`,
                           `psettings`,
                           `menuType`,
                           `type`,
                           `parent`,
                           `start`,
                           `end`,
                           `created`,
                           `modified`

                        ) VALUES (
                            :name,
                            :details,
                            :clientId,
                            :hourBudget,
                            :dollarBudget,
                            :psettings,
                            :menuType,
                            :type,
                            :parent,
                            :start,
                            :end,
                            :created,
                            :modified
                        )';

        $stmn = $this->db->database->prepare($query);

        $stmn->bindValue('name', $values['name'], PDO::PARAM_STR);
        $stmn->bindValue('details', $values['details'], PDO::PARAM_STR);
        $stmn->bindValue('clientId', $values['clientId'], PDO::PARAM_STR);
        $stmn->bindValue('hourBudget', $values['hourBudget'], PDO::PARAM_STR);
        $stmn->bindValue('dollarBudget', $values['dollarBudget'], PDO::PARAM_STR);
        $stmn->bindValue('psettings', $values['psettings'], PDO::PARAM_STR);
        $stmn->bindValue('menuType', $values['menuType'] ?? '', PDO::PARAM_STR);
        $stmn->bindValue('type', $values['type'] ?? 'project', PDO::PARAM_STR);
        $stmn->bindValue('parent', $values['parent'] ?? null, PDO::PARAM_STR);
        $stmn->bindValue('created', date('Y-m-d H:i:s'), PDO::PARAM_STR);
        $stmn->bindValue('modified', date('Y-m-d H:i:s'), PDO::PARAM_STR);

        $startDate = null;
        if (isset($values['start']) && $values['start'] !== false && $values['start'] != '') {
            $startDate = $values['start'];
        }
        $stmn->bindValue('start', $startDate, PDO::PARAM_STR);

        $endDate = null;
        if (isset($values['end']) && $values['end'] !== false && $values['end'] != '') {
            $endDate = $values['end'];
        }

        $stmn->bindValue('end', $endDate, PDO::PARAM_STR);
        $stuff = $stmn->execute();

        $projectId = $this->db->database->lastInsertId();
        $stmn->closeCursor();

        // Add author to project
        if (session()->exists('userdata.id')) {
            $this->addProjectRelation(session('userdata.id'), $projectId, '');
        }

        // Add users to relation
        if (is_array($values['assignedUsers']) === true && count($values['assignedUsers']) > 0) {
            foreach ($values['assignedUsers'] as $user) {
                if (is_array($user) && isset($user['id']) && isset($user['projectRole'])) {
                    $this->addProjectRelation($user['id'], $projectId, $user['projectRole']);
                }
            }
        }

        return is_numeric($projectId) ? (int) $projectId : false;
    }

    /**
     * editProject - edit a project
     */
    public function editProject(array $values, $id): void
    {

        $oldProject = $this->getProject($id);

        $query = 'UPDATE zp_projects SET
				name = :name,
				details = :details,
				clientId = :clientId,
				state = :state,
				hourBudget = :hourBudget,
				dollarBudget = :dollarBudget,
				psettings = :psettings,
				menuType = :menuType,
				type = :type,
				parent = :parent,
				start = :start,
				end = :end,
				modified = :modified
				WHERE id = :id

				LIMIT 1';

        $stmn = $this->db->database->prepare($query);

        $stmn->bindValue('name', $values['name'], PDO::PARAM_STR);
        $stmn->bindValue('details', $values['details'] ?? '', PDO::PARAM_STR);
        $stmn->bindValue('clientId', $values['clientId'] ?? '', PDO::PARAM_STR);
        $stmn->bindValue('state', $values['state'] ?? '', PDO::PARAM_STR);
        $stmn->bindValue('hourBudget', $values['hourBudget'] ?? '', PDO::PARAM_STR);
        $stmn->bindValue('dollarBudget', $values['dollarBudget'] ?? '', PDO::PARAM_STR);
        $stmn->bindValue('psettings', $values['psettings'] ?? '', PDO::PARAM_STR);
        $stmn->bindValue('menuType', $values['menuType'] ?? 'default', PDO::PARAM_STR);
        $stmn->bindValue('type', $values['type'] ?? 'project', PDO::PARAM_STR);
        $stmn->bindValue('id', $id, PDO::PARAM_STR);
        $stmn->bindValue('parent', $values['parent'] ?? null, PDO::PARAM_STR);
        $stmn->bindValue('modified', date('Y-m-d H:i:s'), PDO::PARAM_STR);

        $startDate = null;
        if (isset($values['start']) && $values['start'] !== false && $values['start'] != '') {
            $startDate = $values['start'];
        }
        $stmn->bindValue('start', $startDate, PDO::PARAM_STR);

        $endDate = null;
        if (isset($values['end']) && $values['end'] !== false && $values['end'] != '') {
            $endDate = $values['end'];
        }
        $stmn->bindValue('end', $endDate, PDO::PARAM_STR);

        $stmn->execute();

        $stmn->closeCursor();

        static::dispatch_event('editProject', ['values' => $values, 'oldProject' => $oldProject]);
    }

    /**
     * editProject - edit a project
     */
    public function editProjectRelations(array $values, $projectId): void
    {

        $this->deleteAllUserRelations($projectId);

        // Add users to relation
        if (is_array($values['assignedUsers']) === true && count($values['assignedUsers']) > 0) {
            foreach ($values['assignedUsers'] as $userId) {
                $projectRole = null;
                if (isset($values['projectRoles']['userProjectRole-'.$userId]) && $values['projectRoles']['userProjectRole-'.$userId] != '40' && $values['projectRoles']['userProjectRole-'.$userId] != '50') {
                    $projectRole = (int) $values['projectRoles']['userProjectRole-'.$userId];
                }

                $this->addProjectRelation($userId, $projectId, $projectRole);
            }
        }
    }

    /**
     * deleteProject - delete a project
     */
    public function deleteProject($id): void
    {

        $query = 'DELETE FROM zp_projects WHERE id = :id LIMIT 1';

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':id', $id, PDO::PARAM_INT);

        $stmn->execute();
        $stmn->closeCursor();

        $query = 'DELETE FROM zp_tickets WHERE projectId = :id';

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':id', $id, PDO::PARAM_INT);

        $stmn->execute();
        $stmn->closeCursor();
    }

    /**
     * hasTickets - check if there are Tickets related to a project
     */
    public function hasTickets($id): bool
    {

        $query = "SELECT id FROM zp_tickets WHERE projectId = :id
                      AND zp_tickets.type <> 'subtask' AND
                       zp_tickets.type <> 'milestone' LIMIT 1";

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':id', $id, PDO::PARAM_INT);

        $stmn->execute();
        $values = $stmn->fetchAll();
        $stmn->closeCursor();

        if (count($values) == 0) {
            return false;
        } else {
            return true;
        }
    }

    /**
     * getUserProjectRelation - get all projects related to a user
     *
     * @param  null  $projectId
     */
    public function getUserProjectRelation($id, $projectId = null): array
    {

        $query = 'SELECT
				zp_relationuserproject.userId,
				zp_relationuserproject.projectId,
				zp_projects.name,
				zp_projects.modified,
				zp_relationuserproject.projectRole
			FROM zp_relationuserproject JOIN zp_projects
				ON zp_relationuserproject.projectId = zp_projects.id
			WHERE userId = :id';

        if ($projectId != null) {
            $query .= ' AND  zp_projects.id = :projectId';
        }

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':id', $id, PDO::PARAM_INT);

        if ($projectId != null) {
            $stmn->bindValue(':projectId', $projectId, PDO::PARAM_INT);
        }

        $stmn->execute();
        $values = $stmn->fetchAll();
        $stmn->closeCursor();

        return $values;
    }

    /**
     * @throws BindingResolutionException
     */
    /**
     * @throws BindingResolutionException
     */
    public function isUserAssignedToProject($userId, $projectId): bool
    {

        $userRepo = app()->make(UserRepository::class);
        $user = $userRepo->getUser($userId);

        if ($user === false) {
            return false;
        }

        // admins owners and managers can access everything
        if (in_array(Roles::getRoleString($user['role']), [Roles::$admin, Roles::$owner])) {
            return true;
        }

        $project = $this->getProject($projectId);

        if ($project === false) {
            return false;
        }

        // Everyone in org is allowed to see the project
        if ($project['psettings'] == 'all') {
            return true;
        }

        // Everyone in client is allowed to see project
        if ($project['psettings'] === 'clients') {
            if ($user['clientId'] == $project['clientId']) {
                return true;
            }
        }

        // Select users are allowed to see project
        $query = 'SELECT
				zp_relationuserproject.userId,
				zp_relationuserproject.projectId,
				zp_projects.name,
                zp_projects.modified
			FROM zp_relationuserproject JOIN zp_projects
				ON zp_relationuserproject.projectId = zp_projects.id
			WHERE userId = :userId AND zp_relationuserproject.projectId = :projectId LIMIT 1';

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':userId', $userId, PDO::PARAM_INT);
        $stmn->bindValue(':projectId', $projectId, PDO::PARAM_INT);

        $stmn->execute();
        $values = $stmn->fetch();
        $stmn->closeCursor();

        if ($values && count($values) > 1) {
            return true;
        }

        return false;
    }

    /**
     * @throws BindingResolutionException
     */
    /**
     * @throws BindingResolutionException
     */
    public function isUserMemberOfProject($userId, $projectId): bool
    {

        $userRepo = app()->make(UserRepository::class);
        $user = $userRepo->getUser($userId);

        if ($user === false) {
            return false;
        }

        // admins owners and managers can access everything

        $project = $this->getProject($projectId);

        if ($project === false) {
            return false;
        }

        // Select users are allowed to see project
        $query = 'SELECT
				zp_relationuserproject.userId,
				zp_relationuserproject.projectId,
				zp_projects.name,
                zp_projects.modified
			FROM zp_relationuserproject JOIN zp_projects
				ON zp_relationuserproject.projectId = zp_projects.id
			WHERE userId = :userId AND zp_relationuserproject.projectId = :projectId LIMIT 1';

        $stmn = $this->db->database->prepare($query);
        $stmn->bindValue(':userId', $userId, PDO::PARAM_INT);
        $stmn->bindValue(':projectId', $projectId, PDO::PARAM_INT);

        $stmn->execute();
        $values = $stmn->fetch();
        $stmn->closeCursor();

        if ($values && count($values) > 1) {
            return true;
        }

        return false;
    }

    /**
     * getUserProjectRelation - get all projects related to a user
     */
    public function editUserProjectRelations($id, $projects): bool
    {

        $sql = 'SELECT id,userId,projectId,projectRole FROM zp_relationuserproject WHERE userId=:id';

        $stmn = $this->db->database->prepare($sql);

        $stmn->bindValue(':id', $id, PDO::PARAM_INT);

        $stmn->execute();
        $values = $stmn->fetchAll();
        $stmn->closeCursor();

        // Add relations that don't exist
        foreach ($projects as $project) {
            $exists = false;
            if (count($values)) {
                foreach ($values as $value) {
                    if ($project == $value['projectId']) {
                        $exists = true;
                    }
                }
            }
            if (! $exists) {
                $this->addProjectRelation($id, $project, '');
            }
        }

        // Delete relations that were removed in select
        if (count($values)) {
            foreach ($values as $value) {
                if (in_array($value['projectId'], $projects) !== true) {
                    $this->deleteProjectRelation($id, $value['projectId']);
                }
            }
        }

        return true;
    }

    public function deleteProjectRelation($userId, $projectId): void
    {

        $sql = 'DELETE FROM zp_relationuserproject WHERE projectId=:projectId AND userId=:userId';

        $stmn = $this->db->database->prepare($sql);

        $stmn->bindValue(':userId', $userId, PDO::PARAM_INT);
        $stmn->bindValue(':projectId', $projectId, PDO::PARAM_INT);

        $stmn->execute();

        $stmn->closeCursor();
    }

    public function deleteAllProjectRelations($userId): void
    {

        $sql = 'DELETE FROM zp_relationuserproject WHERE userId=:userId';

        $stmn = $this->db->database->prepare($sql);

        $stmn->bindValue(':userId', $userId, PDO::PARAM_INT);

        $stmn->execute();

        $stmn->closeCursor();
    }

    public function deleteAllUserRelations($projectId): void
    {

        $sql = 'DELETE FROM zp_relationuserproject WHERE projectId=:projectId';

        $stmn = $this->db->database->prepare($sql);

        $stmn->bindValue(':projectId', $projectId, PDO::PARAM_INT);

        $stmn->execute();

        $stmn->closeCursor();
    }

    public function addProjectRelation($userId, $projectId, $projectRole): void
    {
        $oldProject = $this->getProject($projectId);

        $sql = 'INSERT INTO zp_relationuserproject (
					userId,
					projectId,
                    projectRole
				) VALUES (
					:userId,
					:projectId,
					:projectRole
				)';

        $stmn = $this->db->database->prepare($sql);

        $stmn->bindValue(':userId', $userId, PDO::PARAM_INT);
        $stmn->bindValue(':projectId', $projectId, PDO::PARAM_INT);
        $stmn->bindValue(':projectRole', $projectRole, PDO::PARAM_STR);

        $stmn->execute();

        $stmn->closeCursor();

        static::dispatch_event('userAddedToProject', ['userId' => $userId, 'projectId' => $projectId, 'projectRole' => $projectRole, 'oldProject' => $oldProject]);
    }

    public function patch($id, $params): bool
    {

        unset($params['act']);

        $sql = 'UPDATE zp_projects SET ';

        foreach ($params as $key => $value) {
            $sql .= ''.DbCore::sanitizeToColumnString($key).'=:'.DbCore::sanitizeToColumnString($key).', ';
        }

        $sql .= 'modified=:modified ';

        $sql .= ' WHERE id=:id LIMIT 1';

        $stmn = $this->db->database->prepare($sql);
        $stmn->bindValue(':id', $id, PDO::PARAM_STR);
        $stmn->bindValue(':modified', date('Y-m-d H:i:s'), PDO::PARAM_STR);

        foreach ($params as $key => $value) {
            $stmn->bindValue(':'.DbCore::sanitizeToColumnString($key), $value, PDO::PARAM_STR);
        }

        $return = $stmn->execute();
        $stmn->closeCursor();

        return $return;
    }

    /**
     * setPicture - set the profile picture for an individual
     *
     * @throws BindingResolutionException
     */
    public function setPicture($fileId, $id): bool
    {

        $sql = 'UPDATE
                        `zp_projects`
                    SET
                        avatar = :fileId,
                        modified = :modified
                    WHERE id = :userId';

        $stmn = $this->db->database->prepare($sql);
        $stmn->bindValue(':fileId', $fileId, PDO::PARAM_INT);
        $stmn->bindValue(':userId', $id, PDO::PARAM_INT);
        $stmn->bindValue(':modified', date('Y-m-d H:i:s'), PDO::PARAM_STR);

        $result = $stmn->execute();
        $stmn->closeCursor();

        return $result;

    }

    /**
     * @return string[]|SVG
     *
     * @throws BindingResolutionException
     */
    /**
     * @return array|SVG
     *
     * @throws BindingResolutionException
     */
    public function getProjectAvatar($id): array|false
    {

        $value = false;

        if ($id !== false) {
            $sql = 'SELECT avatar, name FROM `zp_projects` WHERE id = :id LIMIT 1';

            $stmn = $this->db->database->prepare($sql);
            $stmn->bindValue(':id', $id, PDO::PARAM_INT);

            $stmn->execute();
            $value = $stmn->fetch();
            $stmn->closeCursor();

        }

        return $value;

    }
}