<?php
/*
 * Copyright © 2021 ExacTI Technology Solutions. All rights reserved.
 * GPLv3 General License.
 * https://exacti.com.br
 * Phacil PHP Framework - https://github.com/exacti/phacil-framework
 */

namespace Phacil\Framework;

use Phacil\Framework\Interfaces\Front as frontInterface;

/** 
 * @package Phacil\Framework 
 * @since 1.0.0
 **/
final class Front implements frontInterface {

	/**
	 * 
	 * @var Registry
	 */
	protected $registry;

	/**
	 * 
	 * @var array
	 */
	protected $pre_action = array();

	protected $error;
	
	/**
	 * 
	 * @param Registry $registry 
	 * @return void 
	 */
	public function __construct(Registry $registry) {
		$this->registry = $registry;
	}
	
	
	/**
	 * @param \Phacil\Framework\Interfaces\Action $pre_action 
	 * @return void 
	 */
	public function addPreAction(\Phacil\Framework\Interfaces\Action $pre_action) {
		$this->pre_action[] = $pre_action;
	}
	
  	
  	/**
  	 * @param Action $action 
  	 * @param string $error 
  	 * @return void 
  	 */
  	public function dispatch(\Phacil\Framework\Interfaces\Action $action, $error) {
		$this->error = $error;

		$this->registry->set('route', $action->getRoute());
			
		foreach ($this->pre_action as $pre_action) {
			$result = $this->execute($pre_action);
					
			if ($result) {
				$action = $result;
				
				break;
			}
		}
			
		while ($action) {
			$action = $this->execute($action);
		}
  	}

	/**
	 * @param object $action 
	 * @return \Phacil\Framework\Interfaces\Action|\Phacil\Framework\Interfaces\Controller|null
	 * @throws Exception 
	 */
	private function execute(\Phacil\Framework\Interfaces\Action $action) {
		$file = $action->getFile();
		$class = $action->getClass();
		$classAlt = $action->getClassAlt();
		$method = $action->getMethod();
		$args = $action->getArgs();

		unset($action);

		if ($file && file_exists($file)) {
			require_once($file);

			foreach($classAlt as $classController){
				try {
					if(class_exists($classController)){
						
						$action = $this->callController($this->injectionClass($classController), $method, $args);
						break;
					}
				} catch (\Phacil\Framework\Exception\Throwable $th) {
					throw $th;
				}
			}
			
		} elseif(!$file && isset($classAlt['class']) && !empty($classAlt['class']) && class_exists($classAlt['class'])) {
			try {
				$action = $this->callController($this->injectionClass($classAlt['class']), $method, $args);
			} catch (\Phacil\Framework\Exception\Throwable $th) {
				throw ($th);
			}
		}else {
			$action = new Action($this->error);
			
			$this->error = '';
		}
		
		return $action;
	}

	/**
	 * 
	 * @param \Phacil\Framework\Interfaces\Controller $controller 
	 * @return \Phacil\Framework\Interfaces\Controller|\Phacil\Framework\Interfaces\Action|null 
	 * @throws \Phacil\Framework\Exception 
	 */
	protected function callController(\Phacil\Framework\Interfaces\Controller $controller, $method, $args = array()) {
		try {
			//code...
			if (is_callable(array($controller, $method))) {
				$action = call_user_func_array(array($controller, $method), $args);
			} else {
				$action = new Action($this->error);

				$this->error = '';
				//throw new Exception("The controller can't be loaded", 1);
				new Exception("PHACIL ERROR: Controller class " . get_class($controller) . "->" . $method . "() is not a callable function");
			}
			return $action;
		} catch (\Exception $th) {
			throw new Exception($th->getMessage(), $th->getCode(), $th);
			//throw $th;
		}
	}

	protected function injectionClass($class){
		return $this->registry->injectionClass($class);
	}
}