Context.php 3.7 KB
Newer Older
1 2
<?php
/**
3 4 5
 * @link http://www.yiiframework.com/
 * @copyright Copyright (c) 2008 Yii Software LLC
 * @license http://www.yiiframework.com/license/
6 7
 */

8
namespace yii\apidoc\models;
9 10


11
use phpDocumentor\Reflection\FileReflector;
12 13 14 15 16
use yii\base\Component;
use yii\base\Exception;

class Context extends Component
{
17 18 19
	/**
	 * @var array list of php files that have been added to this context.
	 */
20 21 22 23 24 25 26 27 28 29 30 31 32 33
	public $files = [];
	/**
	 * @var ClassDoc[]
	 */
	public $classes = [];
	/**
	 * @var InterfaceDoc[]
	 */
	public $interfaces = [];
	/**
	 * @var TraitDoc[]
	 */
	public $traits = [];

34

35 36
	public function addFile($fileName)
	{
37 38 39 40 41 42 43 44 45
		if (isset($this->files[$fileName])) {
			return;
		}
		$this->files[$fileName] = $fileName;

		$reflection = new FileReflector($fileName, true);
		$reflection->process();

		foreach($reflection->getClasses() as $class) {
46
			$class = new ClassDoc($class);
47 48 49 50
			$class->sourceFile = $fileName;
			$this->addClass($class);
		}
		foreach($reflection->getInterfaces() as $interface) {
51
			$interface = new InterfaceDoc($interface);
52 53 54 55
			$interface->sourceFile = $fileName;
			$this->addInterface($interface);
		}
		foreach($reflection->getTraits() as $trait) {
56
			$trait = new TraitDoc($trait);
57 58 59
			$trait->sourceFile = $fileName;
			$this->addTrait($trait);
		}
60 61
	}

62 63 64 65
	/**
	 * @param ClassDoc $class
	 * @throws \yii\base\Exception when class is already part of this context
	 */
66 67 68
	public function addClass($class)
	{
		if (isset($this->classes[$class->name])) {
69
			throw new Exception('Duplicate class definition: ' . $class->name . ' in file ' . $class->sourceFile . '.');
70 71 72 73
		}
		$this->classes[$class->name] = $class;
	}

74 75 76 77
	/**
	 * @param InterfaceDoc $interface
	 * @throws \yii\base\Exception when interface is already part of this context
	 */
78 79 80
	public function addInterface($interface)
	{
		if (isset($this->interfaces[$interface->name])) {
81
			throw new Exception('Duplicate interface definition: ' . $interface->name . ' in file ' . $interface->sourceFile);
82 83 84 85
		}
		$this->interfaces[$interface->name] = $interface;
	}

86 87 88 89
	/**
	 * @param TraitDoc $trait
	 * @throws \yii\base\Exception when trait is already part of this context
	 */
90 91 92
	public function addTrait($trait)
	{
		if (isset($this->traits[$trait->name])) {
93
			throw new Exception('Duplicate trait definition: ' . $trait->name . ' in file ' . $trait->sourceFile);
94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
		}
		$this->traits[$trait->name] = $trait;
	}

	public function updateReferences()
	{
		// update all subclass references
		foreach($this->classes as $class) {
			$className = $class->name;
			while (isset($this->classes[$class->parentClass])) {
				$class = $this->classes[$class->parentClass];
				$class->subclasses[] = $className;
			}
		}
		// update interfaces of subclasses
		foreach($this->classes as $class) {
110
			// TODO do the same for events, constants, methods, properties
111 112
			$this->updateSubclassInferfacesTraits($class);
		}
113 114 115 116 117 118 119 120 121 122 123 124 125
		// update implementedBy and usedBy for interfaces and traits
		foreach($this->classes as $class) {
			foreach($class->interfaces as $interface) {
				if (isset($this->interfaces[$interface])) {
					$this->interfaces[$interface]->implementedBy[] = $class->name;
				}
			}
			foreach($class->traits as $trait) {
				if (isset($this->traits[$trait])) {
					$this->traits[$trait]->usedBy[] = $class->name;
				}
			}
		}
126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141
	}

	/**
	 * Add implemented interfaces and used traits to subclasses
	 * @param ClassDoc $class
	 */
	protected function updateSubclassInferfacesTraits($class)
	{
		foreach($class->subclasses as $subclass) {
			$subclass = $this->classes[$subclass];
			$subclass->interfaces = array_unique(array_merge($subclass->interfaces, $class->interfaces));
			$subclass->traits = array_unique(array_merge($subclass->traits, $class->traits));
			$this->updateSubclassInferfacesTraits($subclass);
		}
	}
}