<?php
/**
 * Zend Framework (http://framework.zend.com/)
 *
 * @link      http://github.com/zendframework/zf2 for the canonical source repository
 * @copyright Copyright (c) 2005-2013 Zend Technologies USA Inc. (http://www.zend.com)
 * @license   http://framework.zend.com/license/new-bsd New BSD License
 */

namespace Zend\Mvc\Controller;

use 
Zend\EventManager\EventManagerAwareInterface;
use 
Zend\EventManager\EventManagerInterface;
use 
Zend\Mvc\Exception;
use 
Zend\ServiceManager\AbstractPluginManager;
use 
Zend\ServiceManager\ConfigInterface;
use 
Zend\ServiceManager\ServiceLocatorAwareInterface;
use 
Zend\ServiceManager\ServiceLocatorInterface;
use 
Zend\Stdlib\DispatchableInterface;

/**
 * Manager for loading controllers
 *
 * Does not define any controllers by default, but does add a validator.
 */
class ControllerManager extends AbstractPluginManager
{
    
/**
     * We do not want arbitrary classes instantiated as controllers.
     *
     * @var bool
     */
    
protected $autoAddInvokableClass false;

    
/**
     * Constructor
     *
     * After invoking parent constructor, add an initializer to inject the
     * service manager, event manager, and plugin manager
     *
     * @param  null|ConfigInterface $configuration
     */
    
public function __construct(ConfigInterface $configuration null)
    {
        
parent::__construct($configuration);
        
// Pushing to bottom of stack to ensure this is done last
        
$this->addInitializer(array($this'injectControllerDependencies'), false);
    }

    
/**
     * Inject required dependencies into the controller.
     *
     * @param  DispatchableInterface $controller
     * @param  ServiceLocatorInterface $serviceLocator
     * @return void
     */
    
public function injectControllerDependencies($controllerServiceLocatorInterface $serviceLocator)
    {
        if (!
$controller instanceof DispatchableInterface) {
            return;
        }

        
$parentLocator $serviceLocator->getServiceLocator();

        if (
$controller instanceof ServiceLocatorAwareInterface) {
            
$controller->setServiceLocator($parentLocator->get('Zend\ServiceManager\ServiceLocatorInterface'));
        }

        if (
$controller instanceof EventManagerAwareInterface) {
            
// If we have an event manager composed already, make sure it gets
            // injected with the shared event manager.
            // The AbstractController lazy-instantiates an EM instance, which
            // is why the shared EM injection needs to happen; the conditional
            // will always pass.
            
$events $controller->getEventManager();
            if (!
$events instanceof EventManagerInterface) {
                
$controller->setEventManager($parentLocator->get('EventManager'));
            } else {
                
$events->setSharedManager($parentLocator->get('SharedEventManager'));
            }
        }

        if (
method_exists($controller'setPluginManager')) {
            
$controller->setPluginManager($parentLocator->get('ControllerPluginManager'));
        }
    }

    
/**
     * Validate the plugin
     *
     * Ensure we have a dispatchable.
     *
     * @param  mixed $plugin
     * @return true
     * @throws Exception\InvalidControllerException
     */
    
public function validatePlugin($plugin)
    {
        if (
$plugin instanceof DispatchableInterface) {
            
// we're okay
            
return;
        }

        throw new 
Exception\InvalidControllerException(sprintf(
            
'Controller of type %s is invalid; must implement Zend\Stdlib\DispatchableInterface',
            (
is_object($plugin) ? get_class($plugin) : gettype($plugin))
        ));
    }

    
/**
     * Override: do not use peering service manager to retrieve controller
     *
     * @param  string $name
     * @param  array $options
     * @param  bool $usePeeringServiceManagers
     * @return mixed
     */
    
public function get($name$options = array(), $usePeeringServiceManagers false)
    {
        return 
parent::get($name$options$usePeeringServiceManagers);
    }
}