asset.php 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358
  1. <?php namespace System;
  2. use System\File;
  3. use System\HTML;
  4. class Asset {
  5. /**
  6. * All of the instantiated asset containers.
  7. *
  8. * Asset containers are created through the container method, and are singletons.
  9. *
  10. * @var array
  11. */
  12. public static $containers = array();
  13. /**
  14. * Get an asset container instance.
  15. *
  16. * If no container name is specified, the default container will be returned.
  17. * Containers provide a convenient method of grouping assets while maintaining
  18. * expressive code and a clean API.
  19. *
  20. * <code>
  21. * // Get the default asset container
  22. * $container = Asset::container();
  23. *
  24. * // Get the "footer" asset contanier
  25. * $container = Asset::container('footer');
  26. *
  27. * // Add an asset to the "footer" container
  28. * Asset::container('footer')->add('jquery', 'js/jquery.js');
  29. * </code>
  30. *
  31. * @param string $container
  32. * @return Asset_Container
  33. */
  34. public static function container($container = 'default')
  35. {
  36. if ( ! isset(static::$containers[$container]))
  37. {
  38. static::$containers[$container] = new Asset_Container($container);
  39. }
  40. return static::$containers[$container];
  41. }
  42. /**
  43. * Magic Method for calling methods on the default Asset container.
  44. *
  45. * <code>
  46. * // Add jQuery to the default container
  47. * Asset::script('jquery', 'js/jquery.js');
  48. *
  49. * // Equivalent call using the container method
  50. * Asset::container()->script('jquery', 'js/jquery.js');
  51. * </code>
  52. */
  53. public static function __callStatic($method, $parameters)
  54. {
  55. return call_user_func_array(array(static::container(), $method), $parameters);
  56. }
  57. }
  58. class Asset_Container {
  59. /**
  60. * The asset container name.
  61. *
  62. * @var string
  63. */
  64. public $name;
  65. /**
  66. * All of the registered assets.
  67. *
  68. * @var array
  69. */
  70. public $assets = array();
  71. /**
  72. * Create a new asset container instance.
  73. *
  74. * @param string $name
  75. * @return void
  76. */
  77. public function __construct($name)
  78. {
  79. $this->name = $name;
  80. }
  81. /**
  82. * Add an asset to the container.
  83. *
  84. * The extension of the asset source will be used to determine the type of
  85. * asset being registered (CSS or JavaScript). If you are using a non-standard
  86. * extension, you may use the style or script methods to register assets.
  87. *
  88. * <code>
  89. * // Register a jQuery asset
  90. * Asset::add('jquery', 'js/jquery.js');
  91. * </code>
  92. *
  93. * You may also specify asset dependencies. This will instruct the class to
  94. * only link to the registered asset after its dependencies have been linked.
  95. * For example, you may wish to make jQuery UI dependent on jQuery.
  96. *
  97. * <code>
  98. * // Register jQuery UI as dependent on jQuery
  99. * Asset::add('jquery-ui', 'js/jquery-ui.js', 'jquery');
  100. *
  101. * // Register jQuery UI with multiple dependencies
  102. * Asset::add('jquery-ui', 'js/jquery-ui.js', array('jquery', 'fader'));
  103. * </code>
  104. *
  105. * @param string $name
  106. * @param string $source
  107. * @param array $dependencies
  108. * @param array $attributes
  109. * @return void
  110. */
  111. public function add($name, $source, $dependencies = array(), $attributes = array())
  112. {
  113. // Since assets may contain timestamps to force a refresh, we will strip them
  114. // off to get the "real" filename of the asset.
  115. $segments = explode('?', $source);
  116. $type = (File::extension($segments[0]) == 'css') ? 'style' : 'script';
  117. return call_user_func(array($this, $type), $name, $source, $dependencies, $attributes);
  118. }
  119. /**
  120. * Add CSS to the registered assets.
  121. *
  122. * @param string $name
  123. * @param string $source
  124. * @param array $dependencies
  125. * @param array $attributes
  126. * @return void
  127. * @see add
  128. */
  129. public function style($name, $source, $dependencies = array(), $attributes = array())
  130. {
  131. if ( ! array_key_exists('media', $attributes))
  132. {
  133. $attributes['media'] = 'all';
  134. }
  135. $this->register('style', $name, $source, $dependencies, $attributes);
  136. }
  137. /**
  138. * Add JavaScript to the registered assets.
  139. *
  140. * @param string $name
  141. * @param string $source
  142. * @param array $dependencies
  143. * @param array $attributes
  144. * @return void
  145. * @see add
  146. */
  147. public function script($name, $source, $dependencies = array(), $attributes = array())
  148. {
  149. $this->register('script', $name, $source, $dependencies, $attributes);
  150. }
  151. /**
  152. * Add an asset to the registered assets.
  153. *
  154. * @param string $type
  155. * @param string $name
  156. * @param string $source
  157. * @param array $dependencies
  158. * @param array $attributes
  159. * @return void
  160. */
  161. private function register($type, $name, $source, $dependencies, $attributes)
  162. {
  163. $dependencies = (array) $dependencies;
  164. $this->assets[$type][$name] = compact('source', 'dependencies', 'attributes');
  165. }
  166. /**
  167. * Get the links to all of the registered CSS assets.
  168. *
  169. * @return string
  170. */
  171. public function styles()
  172. {
  173. return $this->get_group('style');
  174. }
  175. /**
  176. * Get the links to all of the registered JavaScript assets.
  177. *
  178. * @return string
  179. */
  180. public function scripts()
  181. {
  182. return $this->get_group('script');
  183. }
  184. /**
  185. * Get all of the registered assets for a given group.
  186. *
  187. * @param string $group
  188. * @return string
  189. */
  190. private function get_group($group)
  191. {
  192. if ( ! isset($this->assets[$group]) or count($this->assets[$group]) == 0) return '';
  193. $assets = '';
  194. foreach ($this->arrange($this->assets[$group]) as $name => $data)
  195. {
  196. $assets .= $this->get_asset($group, $name);
  197. }
  198. return $assets;
  199. }
  200. /**
  201. * Get the link to a single registered CSS asset.
  202. *
  203. * <code>
  204. * echo $container->get_style('common');
  205. * </code>
  206. *
  207. * @param string $name
  208. * @return string
  209. */
  210. public function get_style($name)
  211. {
  212. return $this->get_asset('style', $name);
  213. }
  214. /**
  215. * Get the link to a single registered JavaScript asset.
  216. *
  217. * <code>
  218. * echo $container->get_script('jquery');
  219. * </code>
  220. *
  221. * @param string $name
  222. * @return string
  223. */
  224. public function get_script($name)
  225. {
  226. return $this->get_asset('script', $name);
  227. }
  228. /**
  229. * Get a registered asset.
  230. *
  231. * @param string $group
  232. * @param string $name
  233. * @return string
  234. */
  235. private function get_asset($group, $name)
  236. {
  237. if ( ! isset($this->assets[$group][$name])) return '';
  238. $asset = $this->assets[$group][$name];
  239. return HTML::$group($asset['source'], $asset['attributes']);
  240. }
  241. /**
  242. * Sort and retrieve assets based on their dependencies
  243. *
  244. * @param array $assets
  245. * @return array
  246. */
  247. private function arrange($assets)
  248. {
  249. list($original, $sorted) = array($assets, array());
  250. while (count($assets) > 0)
  251. {
  252. foreach ($assets as $asset => $value)
  253. {
  254. $this->evaluate_asset($asset, $value, $original, $sorted, $assets);
  255. }
  256. }
  257. return $sorted;
  258. }
  259. /**
  260. * Evaluate an asset and its dependencies.
  261. *
  262. * @param string $asset
  263. * @param string $value
  264. * @param array $original
  265. * @param array $sorted
  266. * @param array $assets
  267. * @return void
  268. */
  269. private function evaluate_asset($asset, $value, $original, &$sorted, &$assets)
  270. {
  271. // If the asset has no more dependencies, we can add it to the sorted list
  272. // and remove it from the array of assets. Otherwise, we will not verify
  273. // the asset's dependencies and determine if they have already been sorted.
  274. if (count($assets[$asset]['dependencies']) == 0)
  275. {
  276. $sorted[$asset] = $value;
  277. unset($assets[$asset]);
  278. }
  279. else
  280. {
  281. foreach ($assets[$asset]['dependencies'] as $key => $dependency)
  282. {
  283. if ( ! $this->dependency_is_valid($asset, $dependency, $original, $assets))
  284. {
  285. unset($assets[$asset]['dependencies'][$key]);
  286. continue;
  287. }
  288. // If the dependency has not yet been added to the sorted list, we can not
  289. // remove it from this asset's array of dependencies. We'll try again on
  290. // the next trip through the loop.
  291. if ( ! isset($sorted[$dependency])) continue;
  292. unset($assets[$asset]['dependencies'][$key]);
  293. }
  294. }
  295. }
  296. /**
  297. * Check that a dependency is valid.
  298. *
  299. * @param string $asset
  300. * @param string $dependency
  301. * @param array $original
  302. * @param array $assets
  303. * @return bool
  304. */
  305. private function dependency_is_valid($asset, $dependency, $original, $assets)
  306. {
  307. if ( ! isset($original[$dependency]))
  308. {
  309. return false;
  310. }
  311. elseif ($dependency === $asset)
  312. {
  313. throw new \Exception("Asset [$asset] is dependent on itself.");
  314. }
  315. elseif (isset($assets[$dependency]) and in_array($asset, $assets[$dependency]['dependencies']))
  316. {
  317. throw new \Exception("Assets [$asset] and [$dependency] have a circular dependency.");
  318. }
  319. }
  320. }