container.php 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263
  1. <?php namespace System\Asset;
  2. use System\HTML;
  3. class Container {
  4. /**
  5. * The asset container name.
  6. *
  7. * @var string
  8. */
  9. public $name;
  10. /**
  11. * All of the registered assets.
  12. *
  13. * @var array
  14. */
  15. public $assets = array();
  16. /**
  17. * Create a new asset container instance.
  18. *
  19. * @param string $name
  20. * @return void
  21. */
  22. public function __construct($name)
  23. {
  24. $this->name = $name;
  25. }
  26. /**
  27. * Add an asset to the container.
  28. *
  29. * @param string $name
  30. * @param string $source
  31. * @param array $dependencies
  32. * @param array $attributes
  33. * @return void
  34. */
  35. public function add($name, $source, $dependencies = array(), $attributes = array())
  36. {
  37. return call_user_func(array($this, (\System\File::extension($source) == 'css') ? 'style' : 'script'), $name, $source, $dependencies, $attributes);
  38. }
  39. /**
  40. * Add CSS to the registered assets.
  41. *
  42. * @param string $name
  43. * @param string $source
  44. * @param array $dependencies
  45. * @param array $attributes
  46. * @return void
  47. */
  48. public function style($name, $source, $dependencies = array(), $attributes = array())
  49. {
  50. if ( ! array_key_exists('media', $attributes))
  51. {
  52. $attributes['media'] = 'all';
  53. }
  54. $this->register('style', $name, $source, $dependencies, $attributes);
  55. }
  56. /**
  57. * Add JavaScript to the registered assets.
  58. *
  59. * @param string $name
  60. * @param string $source
  61. * @param array $dependencies
  62. * @param array $attributes
  63. * @return void
  64. */
  65. public function script($name, $source, $dependencies = array(), $attributes = array())
  66. {
  67. $this->register('script', $name, $source, $dependencies, $attributes);
  68. }
  69. /**
  70. * Add an asset to the registered assets.
  71. *
  72. * @param string $type
  73. * @param string $name
  74. * @param string $source
  75. * @param array $dependencies
  76. * @param array $attributes
  77. * @return void
  78. */
  79. private function register($type, $name, $source, $dependencies, $attributes)
  80. {
  81. $dependencies = (array) $dependencies;
  82. $this->assets[$type][$name] = compact('source', 'dependencies', 'attributes');
  83. }
  84. /**
  85. * Get all of the registered CSS assets.
  86. *
  87. * @return string
  88. */
  89. public function styles()
  90. {
  91. return $this->get_group('style');
  92. }
  93. /**
  94. * Get all of the registered JavaScript assets.
  95. *
  96. * @return string
  97. */
  98. public function scripts()
  99. {
  100. return $this->get_group('script');
  101. }
  102. /**
  103. * Get all of the registered assets for a given group.
  104. *
  105. * @param string $group
  106. * @return string
  107. */
  108. private function get_group($group)
  109. {
  110. if ( ! isset($this->assets[$group]) or count($this->assets[$group]) == 0) return '';
  111. $assets = '';
  112. foreach ($this->arrange($this->assets[$group]) as $name => $data)
  113. {
  114. $assets .= $this->get_asset($group, $name);
  115. }
  116. return $assets;
  117. }
  118. /**
  119. * Get a registered CSS asset.
  120. *
  121. * @param string $name
  122. * @return string
  123. */
  124. public function get_style($name)
  125. {
  126. return $this->get_asset('style', $name);
  127. }
  128. /**
  129. * Get a registered JavaScript asset.
  130. *
  131. * @param string $name
  132. * @return string
  133. */
  134. public function get_script($name)
  135. {
  136. return $this->get_asset('script', $name);
  137. }
  138. /**
  139. * Get a registered asset.
  140. *
  141. * @param string $group
  142. * @param string $name
  143. * @return string
  144. */
  145. private function get_asset($group, $name)
  146. {
  147. if ( ! isset($this->assets[$group][$name]))
  148. {
  149. return '';
  150. }
  151. $asset = $this->assets[$group][$name];
  152. return HTML::$group($asset['source'], $asset['attributes']);
  153. }
  154. /**
  155. * Sort and retrieve assets based on their dependencies
  156. *
  157. * @param array $assets
  158. * @return array
  159. */
  160. private function arrange($assets)
  161. {
  162. list($original, $sorted) = array($assets, array());
  163. while (count($assets) > 0)
  164. {
  165. foreach ($assets as $asset => $value)
  166. {
  167. $this->evaluate_asset($asset, $value, $original, $sorted, $assets);
  168. }
  169. }
  170. return $sorted;
  171. }
  172. /**
  173. * Evaluate an asset and its dependencies.
  174. *
  175. * @param string $asset
  176. * @param string $value
  177. * @param array $original
  178. * @param array $sorted
  179. * @param array $assets
  180. * @return void
  181. */
  182. private function evaluate_asset($asset, $value, $original, &$sorted, &$assets)
  183. {
  184. // If the asset has no more dependencies, we can add it to the sorted list
  185. // and remove it from the array of assets. Otherwise, we will not verify
  186. // the asset's dependencies and determine if they have already been sorted.
  187. if (count($assets[$asset]['dependencies']) == 0)
  188. {
  189. $sorted[$asset] = $value;
  190. unset($assets[$asset]);
  191. }
  192. else
  193. {
  194. foreach ($assets[$asset]['dependencies'] as $key => $dependency)
  195. {
  196. if ( ! $this->dependency_is_valid($asset, $dependency, $original, $assets))
  197. {
  198. unset($assets[$asset]['dependencies'][$key]);
  199. continue;
  200. }
  201. // If the dependency has not yet been added to the sorted list, we can not
  202. // remove it from this asset's array of dependencies. We'll try again on
  203. // the next trip through the loop.
  204. if ( ! isset($sorted[$dependency])) continue;
  205. unset($assets[$asset]['dependencies'][$key]);
  206. }
  207. }
  208. }
  209. /**
  210. * Check that a dependency is valid.
  211. *
  212. * @param string $asset
  213. * @param string $dependency
  214. * @param array $original
  215. * @param array $assets
  216. * @return bool
  217. */
  218. private function dependency_is_valid($asset, $dependency, $original, $assets)
  219. {
  220. if ( ! isset($original[$dependency]))
  221. {
  222. return false;
  223. }
  224. elseif ($dependency === $asset)
  225. {
  226. throw new \Exception("Asset [$asset] is dependent on itself.");
  227. }
  228. elseif (isset($assets[$dependency]) and in_array($asset, $assets[$dependency]['dependencies']))
  229. {
  230. throw new \Exception("Assets [$asset] and [$dependency] have a circular dependency.");
  231. }
  232. }
  233. }