<?php declare(strict_types=1);

/*
 * This file is part of the Rapsys BlogBundle package.
 *
 * (c) Raphaël Gertz <symfony@rapsys.eu>
 *
 * For the full copyright and license information, please view the LICENSE
 * file that was distributed with this source code.
 */

namespace Rapsys\BlogBundle\Controller;

use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;

use Rapsys\BlogBundle\Entity\Article;
use Rapsys\BlogBundle\Entity\Keyword;
use Rapsys\BlogBundle\Entity\User;

/**
 * {@inheritdoc}
 */
class UserController extends AbstractController {
	/**
	 * The user index
	 *
	 * Display users
	 *
	 * @param Request $request The request instance
	 * @return Response The rendered view
	 */
	public function index(Request $request): Response {
		//With not enough users
		if (($this->count = $this->doctrine->getRepository(User::class)->findCountAsInt()) < $this->page * $this->limit) {
			//Throw 404
			throw $this->createNotFoundException($this->translator->trans('Unable to find users'));
		}

		//Get users
		if ($this->context['users'] = $this->doctrine->getRepository(User::class)->findAllAsArray($this->page, $this->limit)) {
			//Set modified
			$this->modified = max(array_map(function ($v) { return $v['modified']; }, $this->context['users']));
		//Without keywords
		} else {
			//Set empty modified
			$this->modified = new \DateTime('-1 year');
		}

		//Create response
		$response = new Response();

		//With logged user
		if ($this->checker->isGranted('IS_AUTHENTICATED_REMEMBERED')) {
			//Set last modified
			$response->setLastModified(new \DateTime('-1 year'));

			//Set as private
			$response->setPrivate();
		//Without logged user
		} else {
			//Set etag
			//XXX: only for public to force revalidation by last modified
			$response->setEtag(md5(serialize($this->context['users'])));

			//Set last modified
			$response->setLastModified($this->modified);

			//Set as public
			$response->setPublic();

			//Without role and modification
			if ($response->isNotModified($request)) {
				//Return 304 response
				return $response;
			}
		}

		//Set keywords
		$this->context['head']['keywords'] = implode(
			', ',
			//Use closure to extract each unique article keywords sorted
			(function ($t) {
				//Return array
				$r = [];

				//Iterate on users
				foreach($t as $u) {
					//Non empty articles
					if (!empty($u['articles'])) {
						//Iterate on articles
						foreach($u['articles'] as $a) {
							//Non empty keywords
							if (!empty($a['keywords'])) {
								//Iterate on keywords
								foreach($a['keywords'] as $k) {
									//Set keyword
									$r[$k['title']] = $k['title'];
								}
							}
						}
					}
				}

				//Sort array
				sort($r);

				//Return array
				return $r;
			})($this->context['users'])
		);

		//Set title
		$this->context['title'] = $this->translator->trans('Users list');

		//Set description
		$this->context['description'] = $this->translator->trans('Welcome to raphaël\'s developer diary user listing');

		//Render the view
		return $this->render('@RapsysBlog/user/index.html.twig', $this->context, $response);
	}

	/**
	 * The user view
	 *
	 * Display user, articles and keywords
	 *
	 * @param Request $request The request instance
	 * @param integer $id The user id
	 * @param ?string $slug The user slug
	 * @return Response The rendered view
	 */
	public function view(Request $request, int $id, ?string $slug): Response {
		//Without user
		if (empty($this->context['user'] = $this->doctrine->getRepository(User::class)->findByIdAsArray($id))) {
			//Throw 404
			throw $this->createNotFoundException($this->translator->trans('Unable to find user: %id%', ['%id%' => $id]));
		}

		//With invalid slug
		if ($slug !== $this->context['user']['slug']) {
			//Redirect on correctly spelled user
			return $this->redirectToRoute('rapsys_blog_user_view', ['id' => $this->context['user']['id'], 'slug' => $this->context['user']['slug']], Response::HTTP_MOVED_PERMANENTLY);
		}

		//Set modified
		$this->modified = $this->context['user']['modified'];

		//Create response
		$response = new Response();

		//With logged user
		if ($this->checker->isGranted('IS_AUTHENTICATED_REMEMBERED')) {
			//Set last modified
			$response->setLastModified(new \DateTime('-1 year'));

			//Set as private
			$response->setPrivate();
		//Without logged user
		} else {
			//Set etag
			//XXX: only for public to force revalidation by last modified
			$response->setEtag(md5(serialize($this->context['user'])));

			//Set last modified
			$response->setLastModified($this->modified);

			//Set as public
			$response->setPublic();

			//Without role and modification
			if ($response->isNotModified($request)) {
				//Return 304 response
				return $response;
			}
		}

		//Set keywords
		$this->context['head']['keywords'] = implode(
			', ',
			//Use closure to extract each unique article keywords sorted
			(function ($u) {
				//Return array
				$r = [];

				//Non empty articles
				if (!empty($u['articles'])) {
					//Iterate on articles
					foreach($u['articles'] as $a) {
						//Non empty keywords
						if (!empty($a['keywords'])) {
							//Iterate on keywords
							foreach($a['keywords'] as $k) {
								//Set keyword
								$r[$k['title']] = $k['title'];
							}
						}
					}
				}

				//Sort array
				sort($r);

				//Return array
				return $r;
			})($this->context['user'])
		);

		//Set title
		$this->context['title'] = $this->context['user']['pseudonym'];

		//Set description
		//TODO: Add user creation ? Add a description field ?
		#$this->context['description'] = $this->context['user']['description'];

		//Render the view
		return $this->render('@RapsysBlog/user/view.html.twig', $this->context, $response);
	}
}