<?php
/**
 * @link http://www.yiiframework.com/
 * @copyright Copyright (c) 2008 Yii Software LLC
 * @license http://www.yiiframework.com/license/
 */

namespace yii\helpers;

use Yii;
use yii\base\InvalidParamException;
use yii\web\Controller;

/**
 * BaseUrl provides concrete implementation for [[Url]].
 *
 * Do not use BaseUrl. Use [[Url]] instead.
 *
 * @author Alexander Makarov <sam@rmcreative.ru>
 * @since 2.0
 */
class BaseUrl
{
	/**
	 * Returns URL for a route.
	 *
	 * @param array|string $route route as a string or route and parameters in form of
	 * `['route', 'param1' => 'value1', 'param2' => 'value2']`.
	 *
	 * If there is a controller running, relative routes are recognized:
	 *
	 * - If the route is an empty string, the current [[\yii\web\Controller::route|route]] will be used;
	 * - If the route contains no slashes at all, it is considered to be an action ID
	 *   of the current controller and will be prepended with [[\yii\web\Controller::uniqueId]];
	 * - If the route has no leading slash, it is considered to be a route relative
	 *   to the current module and will be prepended with the module's uniqueId.
	 *
	 * In case there is no controller, [[\yii\web\UrlManager::createUrl()]] will be used.
	 *
	 * @param boolean|string $scheme URI scheme to use:
	 *
	 * - `false`: relative URL. Default behavior.
	 * - `true`: absolute URL with the current scheme.
	 * - string: absolute URL with string value used as scheme.
	 *
	 * @return string the URL for the route
	 * @throws InvalidParamException if the parameter is invalid.
	 */
	public static function toRoute($route, $scheme = false)
	{
		$route = (array)$route;
		if (Yii::$app->controller instanceof Controller) {
			$route[0] = static::getNormalizedRoute($route[0]);
		}
		if ($scheme) {
			if ($scheme === true) {
				$scheme = null;
			}
			$url = Yii::$app->getUrlManager()->createAbsoluteUrl($route, $scheme);
		} else {
			$url = Yii::$app->getUrlManager()->createUrl($route);
		}
		return $url;
	}

	/**
	 * Normalizes route and makes it suitable for UrlManager. Absolute routes are staying as is
	 * while relative routes are converted to absolute ones.
	 *
	 * A relative route is a route without a leading slash, such as "view", "post/view".
	 *
	 * - If the route is an empty string, the current [[\yii\web\Controller::route|route]] will be used;
	 * - If the route contains no slashes at all, it is considered to be an action ID
	 *   of the current controller and will be prepended with [[\yii\web\Controller::uniqueId]];
	 * - If the route has no leading slash, it is considered to be a route relative
	 *   to the current module and will be prepended with the module's uniqueId.
	 *
	 * @param string $route the route. This can be either an absolute route or a relative route.
	 * @return string normalized route suitable for UrlManager
	 */
	private static function getNormalizedRoute($route)
	{
		if (strpos($route, '/') === false) {
			// empty or an action ID
			$route = $route === '' ? Yii::$app->controller->getRoute() : Yii::$app->controller->getUniqueId() . '/' . $route;
		} elseif ($route[0] !== '/') {
			// relative to module
			$route = ltrim(Yii::$app->controller->module->getUniqueId() . '/' . $route, '/');
		}
		return $route;
	}

	/**
	 * Creates a URL specified by the input parameter.
	 *
	 * If the input parameter is
	 *
	 * - an array: the first array element is considered a route, while the rest of the name-value
	 *   pairs are treated as the parameters to be used for URL creation using [[toRoute()]].
	 *   For example: `['post/index', 'page' => 2]`, `['index']`.
	 *   In case there is no controller, [[\yii\web\UrlManager::createUrl()]] will be used.
	 * - an empty string: the currently requested URL will be returned;
	 * - a non-empty string: it will first be processed by [[Yii::getAlias()]]. If the result
	 *   is an absolute URL, it will be returned either without any change or, if scheme was specified, with scheme
	 *   replaced; Otherwise, the result will be prefixed with [[\yii\web\Request::baseUrl]] and returned.

	 *
	 * @param array|string $url the parameter to be used to generate a valid URL
	 * @param boolean|string $scheme URI scheme to use:
	 *
	 * - `false`: relative URL. Default behavior.
	 * - `true`: absolute URL with the current scheme.
	 * - string: absolute URL with string value used as scheme.
	 *
	 * @return string the normalized URL
	 * @throws InvalidParamException if the parameter is invalid.
	 */
	public static function to($url = '', $scheme = false)
	{
		if (is_array($url)) {
			return static::toRoute($url, $scheme);
		} elseif ($url === '') {
			if ($scheme) {
				$url = Yii::$app->getRequest()->getAbsoluteUrl();
			} else {
				$url = Yii::$app->getRequest()->getUrl();
			}
		} else {
			$url = Yii::getAlias($url);
			if (strpos($url, '://') === false) {
				if ($url === '' || ($url[0] !== '/' && $url[0] !== '#' && strncmp($url, './', 2))) {
					$url = Yii::$app->getRequest()->getBaseUrl() . '/' . $url;
				}
				if ($scheme) {
					$url = Yii::$app->getRequest()->getHostInfo() . $url;
				}
			}
		}
		if ($scheme && $scheme !== true) {
			$pos = strpos($url, '://');
			if ($pos !== false) {
				$url = $scheme . substr($url, $pos);
			}
		}
		return $url;
	}

	/**
	 * Remembers the specified URL so that it can be later fetched back.
	 *
	 * @param string $url URL to remember. Default is the currently requested URL.
	 * @param string $name Name to use to remember URL. Defaults to [[\yii\web\User::returnUrlParam]].
	 * @see previous()
	 */
	public static function remember($url = '', $name = null)
	{
		if ($url === '') {
			$url = Yii::$app->getRequest()->getUrl();
		}

		if ($name === null) {
			Yii::$app->getUser()->setReturnUrl($url);
		} else {
			Yii::$app->getSession()->set($name, $url);
		}
	}

	/**
	 * Returns the URL previously [[remember()|remembered]].
	 *
	 * @param string $name Name used to remember URL. Defaults to [[\yii\web\User::returnUrlParam]].
	 * @return string URL, or null if no such URL was remembered before.
	 * @see remember()
	 */
	public static function previous($name = null)
	{
		if ($name === null) {
			return Yii::$app->getUser()->getReturnUrl();
		} else {
			return Yii::$app->getSession()->get($name);
		}
	}

	/**
	 * Returns the canonical URL of the currently requested page.
	 * The canonical URL is constructed using current controller's [[yii\web\Controller::route]] and
	 * [[yii\web\Controller::actionParams]]. You may use the following code in the layout view to add a link tag
	 * about canonical URL:
	 *
	 * ```php
	 * $this->registerLinkTag(['rel' => 'canonical', 'href' => Url::canonical()]);
	 * ```
	 *
	 * @return string the canonical URL of the currently requested page
	 */
	public static function canonical()
	{
		$params = Yii::$app->controller->actionParams;
		$params[0] = Yii::$app->controller->getRoute();
		return Yii::$app->getUrlManager()->createAbsoluteUrl($params);
	}

	/**
	 * Returns the home URL.
	 *
	 * @param boolean|string $scheme URI scheme to use:
	 *
	 * - `false`: relative URL. Default behavior.
	 * - `true`: absolute URL with the current scheme.
	 * - string: absolute URL with string value used as scheme.
	 *
	 * @return string home URL
	 */
	public static function home($scheme = false)
	{
		if ($scheme) {
			$url = Yii::$app->getRequest()->getHostInfo() . Yii::$app->getHomeUrl();
			if ($scheme !== true) {
				$pos = strpos($url, '://');
				$url = $scheme . substr($url, $pos);
			}
		} else {
			$url = Yii::$app->getHomeUrl();
		}
		return $url;
	}
}