<?php

namespace FML\Script\Features;

use FML\Controls\Control;
use FML\Controls\Label;
use FML\Script\Builder;
use FML\Script\Script;
use FML\Script\ScriptInclude;
use FML\Script\ScriptLabel;

/**
 * Script Feature realizing a mechanism for browsing through Pages
 *
 * @author    steeffeen <mail@steeffeen.com>
 * @copyright FancyManiaLinks Copyright © 2017 Steffen Schröder
 * @license   http://www.gnu.org/licenses/ GNU General Public License, Version 3
 */
class Paging extends ScriptFeature
{

    /*
     * Constants
     */
    const VAR_CURRENT_PAGE             = "FML_Paging_CurrentPage";
    const FUNCTION_UPDATE_CURRENT_PAGE = "FML_UpdateCurrentPage";

    /**
     * @var Label $label Page number Label
     */
    protected $label = null;

    /**
     * @var PagingPage[] $pages Pages
     */
    protected $pages = array();

    /**
     * @var PagingButton[] $buttons Paging Buttons
     */
    protected $buttons = array();

    /**
     * @var int $startPageNumber Start Page number
     */
    protected $startPageNumber = null;

    /**
     * @var int $customMaxPageNumber Custom maximum page number
     */
    protected $customMaxPageNumber = null;

    /**
     * @var string $previousChunkAction Previous chunk action name
     */
    protected $previousChunkAction = null;

    /**
     * @var string $nextChunkAction Next chunk action name
     */
    protected $nextChunkAction = null;

    /**
     * @var bool $chunkActionAppendsPageNumber Chunk action appended with Page number
     */
    protected $chunkActionAppendsPageNumber = null;

    /**
     * Construct a new Paging
     *
     * @api
     * @param Label          $label   (optional) Page number Label
     * @param PagingPage[]   $pages   (optional) Pages
     * @param PagingButton[] $buttons (optional) Pageing Buttons
     */
    public function __construct(Label $label = null, array $pages = null, array $buttons = null)
    {
        if ($label) {
            $this->setLabel($label);
        }
        if ($pages) {
            $this->setPages($pages);
        }
        if ($buttons) {
            $this->setButtons($buttons);
        }
    }

    /**
     * Get the Label showing the Page number
     *
     * @api
     * @return Label
     */
    public function getLabel()
    {
        return $this->label;
    }

    /**
     * Set the Label showing the Page number
     *
     * @api
     * @param Label $label Page number Label
     * @return static
     */
    public function setLabel(Label $label)
    {
        $label->checkId();
        $this->label = $label;
        return $this;
    }

    /**
     * Get the Pages
     *
     * @api
     * @return PagingPage[]
     */
    public function getPages()
    {
        return $this->pages;
    }

    /**
     * Add a new Page Control
     *
     * @api
     * @param Control $pageControl Page Control
     * @param string  $pageNumber  (optional) Page number
     * @return static
     */
    public function addPageControl(Control $pageControl, $pageNumber = null)
    {
        if ($pageNumber === null) {
            $pageNumber = count($this->pages) + 1;
        }
        $page = new PagingPage($pageControl, $pageNumber);
        return $this->addPage($page);
    }

    /**
     * Add a new Page
     *
     * @api
     * @param PagingPage $page Page
     * @return static
     */
    public function addPage(PagingPage $page)
    {
        if (!in_array($page, $this->pages, true)) {
            array_push($this->pages, $page);
        }
        return $this;
    }

    /**
     * Add new Pages
     *
     * @api
     * @param PagingPage[] $pages Pages
     * @return static
     */
    public function setPages(array $pages)
    {
        $this->pages = array();
        foreach ($pages as $page) {
            $this->addPage($page);
        }
        return $this;
    }

    /**
     * Get the Buttons
     *
     * @api
     * @return PagingButton[]
     */
    public function getButtons()
    {
        return $this->buttons;
    }

    /**
     * Add a new Button Control to browse through the Pages
     *
     * @api
     * @param Control $buttonControl Button used for browsing
     * @param int     $browseAction  (optional) Number of browsed Pages per click
     * @return static
     */
    public function addButtonControl(Control $buttonControl, $browseAction = null)
    {
        if ($browseAction === null) {
            $buttonCount = count($this->buttons);
            if ($buttonCount % 2 === 0) {
                $browseAction = $buttonCount / 2 + 1;
            } else {
                $browseAction = $buttonCount / -2 - 1;
            }
        }
        $button = new PagingButton($buttonControl, $browseAction);
        return $this->addButton($button);
    }

    /**
     * Add a new Button to browse through Pages
     *
     * @api
     * @param PagingButton $button Paging Button
     * @return static
     */
    public function addButton(PagingButton $button)
    {
        if (!in_array($button, $this->buttons, true)) {
            array_push($this->buttons, $button);
        }
        return $this;
    }

    /**
     * Set the Paging Buttons
     *
     * @api
     * @param PagingButton[] $buttons Paging Buttons
     * @return static
     */
    public function setButtons(array $buttons)
    {
        $this->buttons = array();
        foreach ($buttons as $button) {
            $this->addButton($button);
        }
        return $this;
    }

    /**
     * Get the start Page number
     *
     * @api
     * @return int
     */
    public function getStartPageNumber()
    {
        return $this->startPageNumber;
    }

    /**
     * Set the start Page number
     *
     * @api
     * @param int $startPageNumber Page number to start with
     * @return static
     */
    public function setStartPageNumber($startPageNumber)
    {
        $this->startPageNumber = (int)$startPageNumber;
        return $this;
    }

    /**
     * Get a custom maximum Page number for using chunks
     *
     * @api
     * @return int
     */
    public function getCustomMaxPageNumber()
    {
        return $this->customMaxPageNumber;
    }

    /**
     * Set a custom maximum Page number for using chunks
     *
     * @api
     * @param int $maxPageNumber Custom maximum Page number
     * @return static
     */
    public function setCustomMaxPageNumber($maxPageNumber)
    {
        $this->customMaxPageNumber = (int)$maxPageNumber;
        return $this;
    }

    /**
     * Get the action triggered when the previous chunk is needed
     *
     * @api
     * @return string
     */
    public function getPreviousChunkAction()
    {
        return $this->previousChunkAction;
    }

    /**
     * Set the action triggered when the previous chunk is needed
     *
     * @api
     * @param string $previousChunkAction Triggered action
     * @return static
     */
    public function setPreviousChunkAction($previousChunkAction)
    {
        $this->previousChunkAction = (string)$previousChunkAction;
        return $this;
    }

    /**
     * Get the action triggered when the next chunk is needed
     *
     * @api
     * @return string
     */
    public function getNextChunkAction()
    {
        return $this->nextChunkAction;
    }

    /**
     * Set the action triggered when the next chunk is needed
     *
     * @api
     * @param string $nextChunkAction Triggered action
     * @return static
     */
    public function setNextChunkAction($nextChunkAction)
    {
        $this->nextChunkAction = (string)$nextChunkAction;
        return $this;
    }

    /**
     * Set the actions triggered when another chunk is needed
     *
     * @api
     * @param string $chunkAction Triggered action
     * @return static
     */
    public function setChunkActions($chunkAction)
    {
        return $this->setNextChunkAction($chunkAction)
                    ->setPreviousChunkAction($chunkAction);
    }

    /**
     * Get if the chunk action should append the needed Page number
     *
     * @api
     * @return bool
     */
    public function getChunkActionAppendsPageNumber()
    {
        return $this->chunkActionAppendsPageNumber;
    }

    /**
     * Set if the chunk action should append the needed Page number
     *
     * @api
     * @param bool $appendPageNumber Append the needed Page number
     * @return static
     */
    public function setChunkActionAppendsPageNumber($appendPageNumber)
    {
        $this->chunkActionAppendsPageNumber = (bool)$appendPageNumber;
        return $this;
    }

    /**
     * @see ScriptFeature::prepare()
     */
    public function prepare(Script $script)
    {
        if (empty($this->pages)) {
            return $this;
        }
        $script->setScriptInclude(ScriptInclude::TEXTLIB);

        $currentPageVariable = self::VAR_CURRENT_PAGE;
        $updatePageFunction  = self::FUNCTION_UPDATE_CURRENT_PAGE;

        $minPageNumber   = 1;
        $startPageNumber = (is_int($this->startPageNumber) ? $this->startPageNumber : $minPageNumber);
        $maxPage         = $this->getMaxPage();
        $maxPageNumber   = $this->customMaxPageNumber;
        if (!is_int($maxPageNumber)) {
            $maxPageNumber = $maxPage->getPageNumber();
        }

        $pagingId    = $maxPage->getControl()
                               ->getId(true, true);
        $pagingId    = Builder::escapeText($maxPage->getControl()
                                                   ->getId());
        $pageLabelId = Builder::EMPTY_STRING;
        if ($this->label) {
            $pageLabelId = Builder::escapeText($this->label->getId());
        }

        $pagesArrayText       = $this->getPagesArrayText();
        $pageButtonsArrayText = $this->getPageButtonsArrayText();

        $previousChunkAction          = Builder::escapeText($this->previousChunkAction);
        $nextChunkAction              = Builder::escapeText($this->nextChunkAction);
        $chunkActionAppendsPageNumber = Builder::getBoolean($this->chunkActionAppendsPageNumber);

        // Init
        $initScriptText = "
declare {$currentPageVariable} for This = Integer[Text];
{$currentPageVariable}[{$pagingId}] = {$startPageNumber};
{$updatePageFunction}({$pagingId}, {$pageLabelId}, 0, {$minPageNumber}, {$maxPageNumber}, {$pagesArrayText}, {$previousChunkAction}, {$nextChunkAction}, {$chunkActionAppendsPageNumber});";
        $script->appendGenericScriptLabel(ScriptLabel::ONINIT, $initScriptText, true);

        // MouseClick
        $clickScriptText = "
declare PageButtons = {$pageButtonsArrayText};
if (PageButtons.existskey(Event.Control.ControlId)) {
	declare BrowseAction = PageButtons[Event.Control.ControlId];
	{$updatePageFunction}({$pagingId}, {$pageLabelId}, BrowseAction, {$minPageNumber}, {$maxPageNumber}, {$pagesArrayText}, {$previousChunkAction}, {$nextChunkAction}, {$chunkActionAppendsPageNumber});
}";
        $script->appendGenericScriptLabel(ScriptLabel::MOUSECLICK, $clickScriptText, true);

        // Update function
        $functionText = "
Void {$updatePageFunction}(Text _PagingId, Text _PageLabelId, Integer _BrowseAction, Integer _MinPageNumber, Integer _MaxPageNumber, Text[Integer] _Pages, Text _PreviousChunkAction, Text _NextChunkAction, Boolean _ChunkActionAppendPageNumber) {
	declare {$currentPageVariable} for This = Integer[Text];
	if (!{$currentPageVariable}.existskey(_PagingId)) return;
	declare NewPageNumber = {$currentPageVariable}[_PagingId] + _BrowseAction;
	if (NewPageNumber < _MinPageNumber) {
		NewPageNumber = _MinPageNumber;
	} else if (NewPageNumber > _MaxPageNumber) {
		NewPageNumber = _MaxPageNumber;
	}
	{$currentPageVariable}[_PagingId] = NewPageNumber;
	if (_Pages.existskey(NewPageNumber)) {
        foreach (PageNumber => PageId in _Pages) {
            declare PageControl <=> Page.GetFirstChild(PageId);
            PageControl.Visible = (PageNumber == NewPageNumber);
        }
        if (_PageLabelId != \"\") {
            declare PageLabel <=> (Page.GetFirstChild(_PageLabelId) as CMlLabel);
            PageLabel.Value = NewPageNumber^\"/\"^_MaxPageNumber;
        }
	} else {
		declare Text ChunkAction;
		if (_BrowseAction < 0) {
			ChunkAction = _PreviousChunkAction;
		} else {
			ChunkAction = _NextChunkAction;
		}
		if (_ChunkActionAppendPageNumber) {
			ChunkAction ^= NewPageNumber;
		}
		TriggerPageAction(ChunkAction);
	}
}";
        $script->addScriptFunction($updatePageFunction, $functionText);
        return $this;
    }

    /**
     * Get the minimum Page
     *
     * @return PagingPage
     */
    protected function getMinPage()
    {
        $minPageNumber = null;
        $minPage       = null;
        foreach ($this->pages as $page) {
            $pageNumber = $page->getPageNumber();
            if ($minPageNumber === null || $pageNumber < $minPageNumber) {
                $minPageNumber = $pageNumber;
                $minPage       = $page;
            }
        }
        return $minPage;
    }

    /**
     * Get the maximum Page
     *
     * @return PagingPage
     */
    protected function getMaxPage()
    {
        $maxPageNumber = null;
        $maxPage       = null;
        foreach ($this->pages as $page) {
            $pageNumber = $page->getPageNumber();
            if ($maxPageNumber === null || $pageNumber > $maxPageNumber) {
                $maxPageNumber = $pageNumber;
                $maxPage       = $page;
            }
        }
        return $maxPage;
    }

    /**
     * Build the array text for the Pages
     *
     * @return string
     */
    protected function getPagesArrayText()
    {
        if (empty($this->pages)) {
            return Builder::getArray(array(0 => ''), true);
        }
        $pages = array();
        foreach ($this->pages as $page) {
            $pages[$page->getPageNumber()] = $page->getControl()
                                                  ->getId();
        }
        return Builder::getArray($pages, true);
    }

    /**
     * Build the array text for the Page Buttons
     *
     * @return string
     */
    protected function getPageButtonsArrayText()
    {
        if (empty($this->buttons)) {
            return Builder::getArray(array('' => 0), true);
        }
        $pageButtons = array();
        foreach ($this->buttons as $pageButton) {
            $pageButtons[$pageButton->getControl()
                                    ->getId()] = $pageButton->getPagingCount();
        }
        return Builder::getArray($pageButtons, true);
    }

}