2011-03-18 17:28:11 +11:00
|
|
|
<?php
|
2016-08-19 10:51:35 +12:00
|
|
|
|
|
|
|
namespace SilverStripe\Core\Manifest;
|
|
|
|
|
|
|
|
use SilverStripe\Core\ClassInfo;
|
|
|
|
use SilverStripe\Dev\Deprecation;
|
|
|
|
|
2011-03-18 17:28:11 +11:00
|
|
|
/**
|
|
|
|
* A class that handles loading classes and interfaces from a class manifest
|
|
|
|
* instance.
|
|
|
|
*/
|
2016-11-29 12:31:16 +13:00
|
|
|
class ClassLoader
|
|
|
|
{
|
2011-03-18 17:28:11 +11:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
2017-03-02 17:19:44 +13:00
|
|
|
* @internal
|
2016-11-29 12:31:16 +13:00
|
|
|
* @var ClassLoader
|
|
|
|
*/
|
|
|
|
private static $instance;
|
2011-03-18 17:28:11 +11:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
2017-06-22 22:50:45 +12:00
|
|
|
* Map of 'instance' (ClassManifest) and other options.
|
|
|
|
*
|
|
|
|
* @var array
|
2016-11-29 12:31:16 +13:00
|
|
|
*/
|
|
|
|
protected $manifests = array();
|
2011-03-18 17:28:11 +11:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* @return ClassLoader
|
|
|
|
*/
|
2017-05-19 14:38:06 +12:00
|
|
|
public static function inst()
|
2016-11-29 12:31:16 +13:00
|
|
|
{
|
2017-06-22 22:50:45 +12:00
|
|
|
return self::$instance ? self::$instance : self::$instance = new static();
|
2016-11-29 12:31:16 +13:00
|
|
|
}
|
2011-03-18 17:28:11 +11:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* Returns the currently active class manifest instance that is used for
|
|
|
|
* loading classes.
|
|
|
|
*
|
|
|
|
* @return ClassManifest
|
|
|
|
*/
|
|
|
|
public function getManifest()
|
|
|
|
{
|
|
|
|
return $this->manifests[count($this->manifests) - 1]['instance'];
|
|
|
|
}
|
2014-08-15 18:53:05 +12:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* Returns true if this class loader has a manifest.
|
|
|
|
*/
|
|
|
|
public function hasManifest()
|
|
|
|
{
|
|
|
|
return (bool)$this->manifests;
|
|
|
|
}
|
2011-03-18 17:28:11 +11:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* Pushes a class manifest instance onto the top of the stack.
|
|
|
|
*
|
|
|
|
* @param ClassManifest $manifest
|
|
|
|
* @param bool $exclusive Marks the manifest as exclusive. If set to FALSE, will
|
|
|
|
* look for classes in earlier manifests as well.
|
|
|
|
*/
|
|
|
|
public function pushManifest(ClassManifest $manifest, $exclusive = true)
|
|
|
|
{
|
|
|
|
$this->manifests[] = array('exclusive' => $exclusive, 'instance' => $manifest);
|
|
|
|
}
|
2011-03-18 17:28:11 +11:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* @return ClassManifest
|
|
|
|
*/
|
|
|
|
public function popManifest()
|
|
|
|
{
|
|
|
|
$manifest = array_pop($this->manifests);
|
|
|
|
return $manifest['instance'];
|
|
|
|
}
|
2011-03-18 17:28:11 +11:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
public function registerAutoloader()
|
|
|
|
{
|
|
|
|
spl_autoload_register(array($this, 'loadClass'));
|
|
|
|
}
|
2011-03-18 17:28:11 +11:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* Loads a class or interface if it is present in the currently active
|
|
|
|
* manifest.
|
|
|
|
*
|
|
|
|
* @param string $class
|
|
|
|
* @return String
|
|
|
|
*/
|
|
|
|
public function loadClass($class)
|
|
|
|
{
|
|
|
|
if ($path = $this->getItemPath($class)) {
|
|
|
|
require_once $path;
|
|
|
|
}
|
|
|
|
return $path;
|
|
|
|
}
|
2014-08-15 18:53:05 +12:00
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* Returns the path for a class or interface in the currently active manifest,
|
|
|
|
* or any previous ones if later manifests aren't set to "exclusive".
|
|
|
|
*
|
|
|
|
* @param string $class
|
|
|
|
* @return string|false
|
|
|
|
*/
|
|
|
|
public function getItemPath($class)
|
|
|
|
{
|
|
|
|
foreach (array_reverse($this->manifests) as $manifest) {
|
|
|
|
/** @var ClassManifest $manifestInst */
|
|
|
|
$manifestInst = $manifest['instance'];
|
|
|
|
if ($path = $manifestInst->getItemPath($class)) {
|
|
|
|
return $path;
|
|
|
|
}
|
|
|
|
if ($manifest['exclusive']) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2011-03-18 17:28:11 +11:00
|
|
|
|
2017-06-22 22:50:45 +12:00
|
|
|
/**
|
|
|
|
* Initialise the class loader
|
|
|
|
*
|
|
|
|
* @param bool $includeTests
|
|
|
|
* @param bool $forceRegen
|
|
|
|
*/
|
|
|
|
public function init($includeTests = false, $forceRegen = false)
|
|
|
|
{
|
|
|
|
foreach ($this->manifests as $manifest) {
|
|
|
|
/** @var ClassManifest $instance */
|
|
|
|
$instance = $manifest['instance'];
|
|
|
|
$instance->init($includeTests, $forceRegen);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->registerAutoloader();
|
|
|
|
}
|
|
|
|
|
2016-11-29 12:31:16 +13:00
|
|
|
/**
|
|
|
|
* Returns true if a class or interface name exists in the manifest.
|
|
|
|
*
|
|
|
|
* @param string $class
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function classExists($class)
|
|
|
|
{
|
|
|
|
Deprecation::notice('4.0', 'Use ClassInfo::exists.');
|
|
|
|
return ClassInfo::exists($class);
|
|
|
|
}
|
2012-03-27 17:04:11 +13:00
|
|
|
}
|