bundle.php 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. <?php namespace Laravel; defined('DS') or die('No direct script access.');
  2. class Bundle {
  3. /**
  4. * All of the application's bundles.
  5. *
  6. * @var array
  7. */
  8. public static $bundles = array();
  9. /**
  10. * A cache of the parsed bundle elements.
  11. *
  12. * @var array
  13. */
  14. public static $elements = array();
  15. /**
  16. * All of the bundles that have been started.
  17. *
  18. * @var array
  19. */
  20. public static $started = array();
  21. /**
  22. * Register a bundle for the application.
  23. *
  24. * @param string $bundle
  25. * @param mixed $config Array of 'location', 'handles' and 'auto'; or string of location.
  26. * @return void
  27. */
  28. public static function register($bundle, $config = array())
  29. {
  30. $defaults = array('handles' => null, 'auto' => false);
  31. // If the given config is actually a string, we will assume it is a location
  32. // and convert it to an array so that the developer may conveniently add
  33. // bundles to the configuration without making an array for each one.
  34. if (is_string($config))
  35. {
  36. $config = array('location' => $config);
  37. }
  38. if ( ! isset($config['location']))
  39. {
  40. $config['location'] = $bundle;
  41. }
  42. // We will trim the trailing slash from the location and add it back so
  43. // we don't have to worry about the developer adding or not adding it
  44. // to the location path for the bundle.
  45. $config['location'] = path('bundle').rtrim($config['location'], DS).DS;
  46. // If the handles clause is set, we will append a trailing slash so
  47. // that it is not ultra-greedy. Otherwise, bundles that handle "s"
  48. // would handle all bundles that start with "s".
  49. if (isset($config['handles']))
  50. {
  51. $config['handles'] = $config['handles'].'/';
  52. }
  53. static::$bundles[$bundle] = array_merge($defaults, $config);
  54. }
  55. /**
  56. * Load a bundle by running it's start-up script.
  57. *
  58. * If the bundle has already been started, no action will be taken.
  59. *
  60. * @param string $bundle
  61. * @return void
  62. */
  63. public static function start($bundle)
  64. {
  65. if (static::started($bundle)) return;
  66. if ($bundle !== DEFAULT_BUNDLE and ! static::exists($bundle))
  67. {
  68. throw new \Exception("Bundle [$bundle] has not been installed.");
  69. }
  70. // Each bundle may have a "start" script which is responsible for preparing
  71. // the bundle for use by the application. The start script may register any
  72. // classes the bundle uses with the auto-loader, or perhaps will start any
  73. // dependent bundles so that they are available.
  74. if (file_exists($path = static::path($bundle).'bundle'.EXT))
  75. {
  76. require $path;
  77. }
  78. // Each bundle may also have a "routes" file which is responsible for
  79. // registering the bundle's routes. This is kept separate from the
  80. // start script for reverse routing efficiency purposes.
  81. static::routes($bundle);
  82. Event::fire("started: {$bundle}");
  83. static::$started[] = strtolower($bundle);
  84. }
  85. /**
  86. * Load the "routes" file for a given bundle.
  87. *
  88. * @param string $bundle
  89. * @return void
  90. */
  91. public static function routes($bundle)
  92. {
  93. if (file_exists($path = static::path($bundle).'routes'.EXT))
  94. {
  95. require $path;
  96. }
  97. }
  98. /**
  99. * Determine which bundle handles the given URI.
  100. *
  101. * If no bundle is assigned to handle the URI, the default bundle is returned.
  102. *
  103. * @param string $uri
  104. * @return string
  105. */
  106. public static function handles($uri)
  107. {
  108. $uri = rtrim($uri, '/').'/';
  109. foreach (static::$bundles as $key => $value)
  110. {
  111. if (starts_with($uri, $value['handles'])) return $key;
  112. }
  113. return DEFAULT_BUNDLE;
  114. }
  115. /**
  116. * Deteremine if a bundle exists within the bundles directory.
  117. *
  118. * @param string $bundle
  119. * @return bool
  120. */
  121. public static function exists($bundle)
  122. {
  123. return $bundle == DEFAULT_BUNDLE or in_array(strtolower($bundle), static::names());
  124. }
  125. /**
  126. * Determine if a given bundle has been started for the request.
  127. *
  128. * @param string $bundle
  129. * @return void
  130. */
  131. public static function started($bundle)
  132. {
  133. return in_array(strtolower($bundle), static::$started);
  134. }
  135. /**
  136. * Get the identifier prefix for the bundle.
  137. *
  138. * @param string $bundle
  139. * @return string
  140. */
  141. public static function prefix($bundle)
  142. {
  143. return ($bundle !== DEFAULT_BUNDLE) ? "{$bundle}::" : '';
  144. }
  145. /**
  146. * Get the class prefix for a given bundle.
  147. *
  148. * @param string $bundle
  149. * @return string
  150. */
  151. public static function class_prefix($bundle)
  152. {
  153. return ($bundle !== DEFAULT_BUNDLE) ? Str::classify($bundle).'_' : '';
  154. }
  155. /**
  156. * Return the root bundle path for a given bundle.
  157. *
  158. * <code>
  159. * // Returns the bundle path for the "admin" bundle
  160. * $path = Bundle::path('admin');
  161. *
  162. * // Returns the path('app') constant as the default bundle
  163. * $path = Bundle::path('application');
  164. * </code>
  165. *
  166. * @param string $bundle
  167. * @return string
  168. */
  169. public static function path($bundle)
  170. {
  171. return ($bundle == DEFAULT_BUNDLE) ? path('app') : static::$bundles[$bundle]['location'];
  172. }
  173. /**
  174. * Return the root asset path for the given bundle.
  175. *
  176. * @param string $bundle
  177. * @return string
  178. */
  179. public static function assets($bundle)
  180. {
  181. return ($bundle != DEFAULT_BUNDLE) ? URL::base()."/bundles/{$bundle}/" : URL::base().'/';
  182. }
  183. /**
  184. * Get the bundle name from a given identifier.
  185. *
  186. * <code>
  187. * // Returns "admin" as the bundle name for the identifier
  188. * $bundle = Bundle::name('admin::home.index');
  189. * </code>
  190. *
  191. * @param string $identifier
  192. * @return string
  193. */
  194. public static function name($identifier)
  195. {
  196. list($bundle, $element) = static::parse($identifier);
  197. return $bundle;
  198. }
  199. /**
  200. * Get the element name from a given identifier.
  201. *
  202. * <code>
  203. * // Returns "home.index" as the element name for the identifier
  204. * $bundle = Bundle::bundle('admin::home.index');
  205. * </code>
  206. *
  207. * @param string $identifier
  208. * @return string
  209. */
  210. public static function element($identifier)
  211. {
  212. list($bundle, $element) = static::parse($identifier);
  213. return $element;
  214. }
  215. /**
  216. * Reconstruct an identifier from a given bundle and element.
  217. *
  218. * <code>
  219. * // Returns "admin::home.index"
  220. * $identifier = Bundle::identifier('admin', 'home.index');
  221. *
  222. * // Returns "home.index"
  223. * $identifier = Bundle::identifier('application', 'home.index');
  224. * </code>
  225. *
  226. * @param string $bundle
  227. * @param string $element
  228. * @return string
  229. */
  230. public static function identifier($bundle, $element)
  231. {
  232. return (is_null($bundle) or $bundle == DEFAULT_BUNDLE) ? $element : $bundle.'::'.$element;
  233. }
  234. /**
  235. * Return the bundle name if it exists, else return the default bundle.
  236. *
  237. * @param string $bundle
  238. * @return string
  239. */
  240. public static function resolve($bundle)
  241. {
  242. return (static::exists($bundle)) ? $bundle : DEFAULT_BUNDLE;
  243. }
  244. /**
  245. * Parse a element identifier and return the bundle name and element.
  246. *
  247. * <code>
  248. * // Returns array(null, 'admin.user')
  249. * $element = Bundle::parse('admin.user');
  250. *
  251. * // Parses "admin::user" and returns array('admin', 'user')
  252. * $element = Bundle::parse('admin::user');
  253. * </code>
  254. *
  255. * @param string $identifier
  256. * @return array
  257. */
  258. public static function parse($identifier)
  259. {
  260. // The parsed elements are cached so we don't have to reparse them on each
  261. // subsequent request for the parsed element. So, if we've already parsed
  262. // the given element, we'll just return the cached copy.
  263. if (isset(static::$elements[$identifier]))
  264. {
  265. return static::$elements[$identifier];
  266. }
  267. if (strpos($identifier, '::') !== false)
  268. {
  269. $element = explode('::', strtolower($identifier));
  270. }
  271. // If no bundle is in the identifier, we will insert the default bundle
  272. // since classes like Config and Lang organize their items by bundle.
  273. // The "application" folder essentially behaves as a bundle.
  274. else
  275. {
  276. $element = array(DEFAULT_BUNDLE, strtolower($identifier));
  277. }
  278. return static::$elements[$identifier] = $element;
  279. }
  280. /**
  281. * Get the information for a given bundle.
  282. *
  283. * @param string $bundle
  284. * @return object
  285. */
  286. public static function get($bundle)
  287. {
  288. return (object) array_get(static::$bundles, $bundle);
  289. }
  290. /**
  291. * Get all of the installed bundles for the application.
  292. *
  293. * @return array
  294. */
  295. public static function all()
  296. {
  297. return static::$bundles;
  298. }
  299. /**
  300. * Get all of the installed bundle names.
  301. *
  302. * @return array
  303. */
  304. public static function names()
  305. {
  306. return array_keys(static::$bundles);
  307. }
  308. }