view.php 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376
  1. <?php namespace Laravel; use Closure;
  2. class View_Factory {
  3. /**
  4. * The directory containing the views.
  5. *
  6. * @var string
  7. */
  8. public $path;
  9. /**
  10. * The path to the directory containing compiled views.
  11. *
  12. * @var string
  13. */
  14. public $compiled;
  15. /**
  16. * The view composer instance.
  17. *
  18. * @var View_Composer
  19. */
  20. protected $composer;
  21. /**
  22. * Create a new view factory instance.
  23. *
  24. * @param View_Composer $composer
  25. * @param string $path
  26. * @param string $compiled
  27. * @return void
  28. */
  29. public function __construct(View_Composer $composer, $path, $compiled)
  30. {
  31. $this->path = $path;
  32. $this->composer = $composer;
  33. $this->compiled = $compiled;
  34. }
  35. /**
  36. * Create a new view instance.
  37. *
  38. * The name of the view given to this method should correspond to a view
  39. * within your application views directory. Dots or slashes may used to
  40. * reference views within sub-directories.
  41. *
  42. * <code>
  43. * // Create a new view instance
  44. * $view = View::make('home.index');
  45. *
  46. * // Create a new view instance with bound data
  47. * $view = View::make('home.index', array('name' => 'Taylor'));
  48. * </code>
  49. *
  50. * @param string $view
  51. * @param array $data
  52. * @return View
  53. */
  54. public function make($view, $data = array())
  55. {
  56. return new View($this, $this->composer, $view, $data, $this->path($view));
  57. }
  58. /**
  59. * Create a new view instance from a view name.
  60. *
  61. * View names are defined in the application composers file.
  62. *
  63. * <code>
  64. * // Create an instance of the "layout" named view
  65. * $view = View::of('layout');
  66. *
  67. * // Create an instance of the "layout" view with bound data
  68. * $view = View::of('layout', array('name' => 'Taylor'));
  69. * </code>
  70. *
  71. * @param string $name
  72. * @param array $data
  73. * @return View
  74. */
  75. public function of($name, $data = array())
  76. {
  77. if ( ! is_null($view = $this->composer->name($name))) return $this->make($view, $data);
  78. throw new \Exception("Named view [$name] is not defined.");
  79. }
  80. /**
  81. * Get the path to a given view on disk.
  82. *
  83. * @param string $view
  84. * @return string
  85. */
  86. protected function path($view)
  87. {
  88. $view = str_replace('.', '/', $view);
  89. foreach (array(EXT, BLADE_EXT) as $extension)
  90. {
  91. if (file_exists($path = $this->path.$view.$extension)) return $path;
  92. }
  93. throw new \Exception("View [$view] does not exist.");
  94. }
  95. /**
  96. * Magic Method for handling the dynamic creation of named views.
  97. *
  98. * <code>
  99. * // Create an instance of the "layout" named view
  100. * $view = View::of_layout();
  101. *
  102. * // Create an instance of a named view with data
  103. * $view = View::of_layout(array('name' => 'Taylor'));
  104. * </code>
  105. */
  106. public function __call($method, $parameters)
  107. {
  108. if (strpos($method, 'of_') === 0)
  109. {
  110. return $this->of(substr($method, 3), Arr::get($parameters, 0, array()));
  111. }
  112. }
  113. }
  114. class View_Composer {
  115. /**
  116. * The view composers.
  117. *
  118. * @var array
  119. */
  120. protected $composers;
  121. /**
  122. * Create a new view composer instance.
  123. *
  124. * @param array $composers
  125. * @return void
  126. */
  127. public function __construct($composers)
  128. {
  129. $this->composers = $composers;
  130. }
  131. /**
  132. * Find the key for a view by name.
  133. *
  134. * The view's key can be used to create instances of the view through the typical
  135. * methods available on the view factory.
  136. *
  137. * @param string $name
  138. * @return string
  139. */
  140. public function name($name)
  141. {
  142. // The view's name may specified in several different ways in the composers file.
  143. // The composer may simple have a string value, which is the name. Or, the composer
  144. // could have an array value in which a "name" key exists.
  145. foreach ($this->composers as $key => $value)
  146. {
  147. if ($name === $value or (isset($value['name']) and $name === $value['name'])) { return $key; }
  148. }
  149. }
  150. /**
  151. * Call the composer for the view instance.
  152. *
  153. * @param View $view
  154. * @return void
  155. */
  156. public function compose(View $view)
  157. {
  158. // The shared composer is called for every view instance. This allows the
  159. // convenient binding of global view data or partials within a single method.
  160. if (isset($this->composers['shared'])) call_user_func($this->composers['shared'], $view);
  161. if (isset($this->composers[$view->view]))
  162. {
  163. foreach ((array) $this->composers[$view->view] as $key => $value)
  164. {
  165. if ($value instanceof Closure) return call_user_func($value, $view);
  166. }
  167. }
  168. }
  169. }
  170. class View {
  171. /**
  172. * The name of the view.
  173. *
  174. * @var string
  175. */
  176. public $view;
  177. /**
  178. * The view data.
  179. *
  180. * @var array
  181. */
  182. public $data;
  183. /**
  184. * The path to the view on disk.
  185. *
  186. * @var string
  187. */
  188. protected $path;
  189. /**
  190. * The view composer instance.
  191. *
  192. * @var View_Composer
  193. */
  194. protected $composer;
  195. /**
  196. * The view factory instance, which is used to create sub-views.
  197. *
  198. * @var View_Factory
  199. */
  200. protected $factory;
  201. /**
  202. * Create a new view instance.
  203. *
  204. * @param View_Factory $factory
  205. * @param View_Composer $composer
  206. * @param string $view
  207. * @param array $data
  208. * @param string $path
  209. * @return void
  210. */
  211. public function __construct(View_Factory $factory, View_Composer $composer, $view, $data, $path)
  212. {
  213. $this->view = $view;
  214. $this->data = $data;
  215. $this->path = $path;
  216. $this->factory = $factory;
  217. $this->composer = $composer;
  218. }
  219. /**
  220. * Get the evaluated string content of the view.
  221. *
  222. * @return string
  223. */
  224. public function render()
  225. {
  226. $this->composer->compose($this);
  227. // All nested views and responses are evaluated before the main view. This allows
  228. // the assets used by these views to be added to the asset container before the
  229. // main view is evaluated and dumps the links to the assets.
  230. foreach ($this->data as &$data)
  231. {
  232. if ($data instanceof View or $data instanceof Response) $data = $data->render();
  233. }
  234. // We don't want the view's contents to be rendered immediately, so we will fire
  235. // up an output buffer to catch the view output. The output of the view will be
  236. // rendered automatically later in the request lifecycle.
  237. ob_start() and extract($this->data, EXTR_SKIP);
  238. // If the view is a "Blade" view, we need to check the view for modifications
  239. // and get the path to the compiled view file. Otherwise, we'll just use the
  240. // regular path to the view.
  241. $view = (strpos($this->path, BLADE_EXT) !== false) ? $this->compile() : $this->path;
  242. try { include $view; } catch (Exception $e) { ob_get_clean(); throw $e; }
  243. return ob_get_clean();
  244. }
  245. /**
  246. * Compile the Bladed view and return the path to the compiled view.
  247. *
  248. * @return string
  249. */
  250. protected function compile()
  251. {
  252. // For simplicity, compiled views are stored in a single directory by the MD5 hash of
  253. // their name. This allows us to avoid recreating the entire view directory structure
  254. // within the compiled views directory.
  255. $compiled = $this->factory->compiled.md5($this->view);
  256. // The view will only be re-compiled if the view has been modified since the last compiled
  257. // version of the view was created or no compiled view exists. Otherwise, the path will
  258. // be returned without re-compiling.
  259. if ((file_exists($compiled) and filemtime($this->path) > filemtime($compiled)) or ! file_exists($compiled))
  260. {
  261. file_put_contents($compiled, Blade::parse($this->path));
  262. }
  263. return $compiled;
  264. }
  265. /**
  266. * Add a view instance to the view data.
  267. *
  268. * <code>
  269. * // Add a view instance to a view's data
  270. * $view = View::make('foo')->nest('footer', 'partials.footer');
  271. *
  272. * // Equivalent functionality using the "with" method
  273. * $view = View::make('foo')->with('footer', View::make('partials.footer'));
  274. *
  275. * // Bind a view instance with data
  276. * $view = View::make('foo')->nest('footer', 'partials.footer', array('name' => 'Taylor'));
  277. * </code>
  278. *
  279. * @param string $key
  280. * @param string $view
  281. * @param array $data
  282. * @return View
  283. */
  284. public function nest($key, $view, $data = array())
  285. {
  286. return $this->with($key, $this->factory->make($view, $data));
  287. }
  288. /**
  289. * Add a key / value pair to the view data.
  290. *
  291. * Bound data will be available to the view as variables.
  292. *
  293. * @param string $key
  294. * @param mixed $value
  295. * @return View
  296. */
  297. public function with($key, $value)
  298. {
  299. $this->data[$key] = $value;
  300. return $this;
  301. }
  302. /**
  303. * Magic Method for getting items from the view data.
  304. */
  305. public function __get($key)
  306. {
  307. return $this->data[$key];
  308. }
  309. /**
  310. * Magic Method for setting items in the view data.
  311. */
  312. public function __set($key, $value)
  313. {
  314. $this->with($key, $value);
  315. }
  316. /**
  317. * Magic Method for determining if an item is in the view data.
  318. */
  319. public function __isset($key)
  320. {
  321. return array_key_exists($key, $this->data);
  322. }
  323. /**
  324. * Magic Method for removing an item from the view data.
  325. */
  326. public function __unset($key)
  327. {
  328. unset($this->data[$key]);
  329. }
  330. }