index.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372
  1. <?php
  2. namespace mindplay\demo;
  3. use Composer\Autoload\ClassLoader;
  4. use mindplay\annotations\AnnotationCache;
  5. use mindplay\annotations\Annotations;
  6. use mindplay\demo\annotations\Package;
  7. ## Configure a simple auto-loader
  8. $vendor_path = dirname(__DIR__) . '/vendor';
  9. if (!is_dir($vendor_path)) {
  10. echo 'Install dependencies first' . PHP_EOL;
  11. exit(1);
  12. }
  13. require_once($vendor_path . '/autoload.php');
  14. $auto_loader = new ClassLoader();
  15. $auto_loader->addPsr4("mindplay\\demo\\", __DIR__);
  16. $auto_loader->register();
  17. ## Configure the cache-path. The static `Annotations` class will configure any public
  18. ## properties of `AnnotationManager` when it creates it. The `AnnotationManager::$cachePath`
  19. ## property is a path to a writable folder, where the `AnnotationManager` caches parsed
  20. ## annotations from individual source code files.
  21. Annotations::$config['cache'] = new AnnotationCache(__DIR__ . '/runtime');
  22. ## Register demo annotations.
  23. Package::register(Annotations::getManager());
  24. ## For this example, we're going to generate a simple form that allows us to edit a `Person`
  25. ## object. We'll define a few public properties and annotate them with some useful metadata,
  26. ## which will enable us to make decisions (at run-time) about how to display each field,
  27. ## how to parse the values posted back from the form, and how to validate the input.
  28. ##
  29. ## Note the use of standard PHP-DOC annotations, such as `@var string` - this metadata is
  30. ## traditionally useful both as documentation to developers, and as hints for an IDE. In
  31. ## this example, we're going to use that same information as advice to our components, at
  32. ## run-time, to help them establish defaults and make sensible decisions about how to
  33. ## handle the value of each property.
  34. class Person
  35. {
  36. /**
  37. * @var string
  38. * @required
  39. * @length(50)
  40. * @text('label' => 'Full Name')
  41. */
  42. public $name;
  43. /**
  44. * @var string
  45. * @length(50)
  46. * @text('label' => 'Street Address')
  47. */
  48. public $address;
  49. /**
  50. * @var int
  51. * @range(0, 100)
  52. */
  53. public $age;
  54. }
  55. ## To build a simple form abstraction that can manage the state of an object being edited,
  56. ## we start with a simple, abstract base class for input widgets.
  57. abstract class Widget
  58. {
  59. protected $object;
  60. protected $property;
  61. public $value;
  62. ## Each widget will maintain a list of error messages.
  63. public $errors = array();
  64. ## A widget needs to know which property of what object is being edited.
  65. public function __construct($object, $property)
  66. {
  67. $this->object = $object;
  68. $this->property = $property;
  69. $this->value = $object->$property;
  70. }
  71. ## Widget classes will use this method to add an error-message.
  72. public function addError($message)
  73. {
  74. $this->errors[] = $message;
  75. }
  76. ## This helper function provides a shortcut to get a named property from a
  77. ## particular type of annotation - if no annotation is found, the `$default`
  78. ## value is returned instead.
  79. protected function getMetadata($type, $name, $default = null)
  80. {
  81. $a = Annotations::ofProperty($this->object, $this->property, $type);
  82. if (!count($a)) {
  83. return $default;
  84. }
  85. return $a[0]->$name;
  86. }
  87. ## Each type of widget will need to implement this interface, which takes a raw
  88. ## POST value from the form, and attempts to bind it to the object's property.
  89. abstract public function update($input);
  90. ## After a widget successfully updates a property, we may need to perform additional
  91. ## validation - this method will perform some basic validations, and if errors are
  92. ## found, it will add them to the `$errors` collection.
  93. public function validate()
  94. {
  95. if (empty($this->value)) {
  96. if ($this->isRequired()) {
  97. $this->addError("Please complete this field");
  98. } else {
  99. return;
  100. }
  101. }
  102. if (is_string($this->value)) {
  103. $min = $this->getMetadata('@length', 'min');
  104. $max = $this->getMetadata('@length', 'max');
  105. if ($min !== null && strlen($this->value) < $min) {
  106. $this->addError("Minimum length is {$min} characters");
  107. } else {
  108. if ($max !== null && strlen($this->value) > $max) {
  109. $this->addError("Maximum length is {$max} characters");
  110. }
  111. }
  112. }
  113. if (is_int($this->value)) {
  114. $min = $this->getMetadata('@range', 'min');
  115. $max = $this->getMetadata('@range', 'max');
  116. if (($min !== null && $this->value < $min) || ($max !== null && $this->value > $max)) {
  117. $this->addError("Please enter a value in the range {$min} to {$max}");
  118. }
  119. }
  120. }
  121. ## Each type of widget will need to implement this interface, which renders an
  122. ## HTML input representing the widget's current value.
  123. abstract public function display();
  124. ## This helper function returns a descriptive label for the input.
  125. public function getLabel()
  126. {
  127. return $this->getMetadata('@text', 'label', ucfirst($this->property));
  128. }
  129. ## Finally, this little helper function will tell us if the field is required -
  130. ## if a property is annotated with `@required`, the field must be filled in.
  131. public function isRequired()
  132. {
  133. return count(Annotations::ofProperty($this->object, $this->property, '@required')) > 0;
  134. }
  135. }
  136. ## The first and most basic kind of widget, is this simple string widget.
  137. class StringWidget extends Widget
  138. {
  139. ## On update, take into account the min/max string length, and provide error
  140. ## messages if the constraints are violated.
  141. public function update($input)
  142. {
  143. $this->value = $input;
  144. $this->validate();
  145. }
  146. ## On display, render out a simple `<input type="text"/>` field, taking into account
  147. ## the maximum string-length.
  148. public function display()
  149. {
  150. $length = $this->getMetadata('@length', 'max', 255);
  151. echo '<input type="text" name="' . get_class($this->object) . '[' . $this->property . ']"'
  152. . ' maxlength="' . $length . '" value="' . htmlspecialchars($this->value) . '"/>';
  153. }
  154. }
  155. ## For the age input, we'll need a specialized `StringWidget` that also checks the input type.
  156. class IntWidget extends StringWidget
  157. {
  158. ## On update, take into account the min/max numerical range, and provide error
  159. ## messages if the constraints are violated.
  160. public function update($input)
  161. {
  162. if (strval(intval($input)) === $input) {
  163. $this->value = intval($input);
  164. $this->validate();
  165. } else {
  166. $this->value = $input;
  167. if (!empty($input)) {
  168. $this->addError("Please enter a whole number value");
  169. }
  170. }
  171. }
  172. }
  173. ## Next, we can build a simple form abstraction - this will hold and object and manage
  174. ## the widgets required to edit the object.
  175. class Form
  176. {
  177. private $object;
  178. /**
  179. * Widget list.
  180. *
  181. * @var Widget[]
  182. */
  183. private $widgets = array();
  184. ## The constructor just needs to know which object we're editing.
  185. ##
  186. ## Using reflection, we enumerate the properties of the object's type, and using the
  187. ## `@var` annotation, we decide which type of widget we're going to use.
  188. public function __construct($object)
  189. {
  190. $this->object = $object;
  191. $class = new \ReflectionClass($this->object);
  192. foreach ($class->getProperties() as $property) {
  193. $type = $this->getMetadata($property->name, '@var', 'type', 'string');
  194. $wtype = 'mindplay\\demo\\' . ucfirst($type) . 'Widget';
  195. $this->widgets[$property->name] = new $wtype($this->object, $property->name);
  196. }
  197. }
  198. ## This helper-method is similar to the one we defined for the widget base
  199. ## class, but fetches annotations for the specified property.
  200. private function getMetadata($property, $type, $name, $default = null)
  201. {
  202. $a = Annotations::ofProperty(get_class($this->object), $property, $type);
  203. if (!count($a)) {
  204. return $default;
  205. }
  206. return $a[0]->$name;
  207. }
  208. ## When you post information back to the form, we'll need to update it's state,
  209. ## validate each of the fields, and return a value indicating whether the form
  210. ## update was successful.
  211. public function update($post)
  212. {
  213. $data = $post[get_class($this->object)];
  214. foreach ($this->widgets as $property => $widget) {
  215. if (array_key_exists($property, $data)) {
  216. $this->widgets[$property]->update($data[$property]);
  217. }
  218. }
  219. $valid = true;
  220. foreach ($this->widgets as $widget) {
  221. $valid = $valid && (count($widget->errors) === 0);
  222. }
  223. if ($valid) {
  224. foreach ($this->widgets as $property => $widget) {
  225. $this->object->$property = $widget->value;
  226. }
  227. }
  228. return $valid;
  229. }
  230. ## Finally, this method renders out the form, and each of the widgets inside, with
  231. ## a `<label>` tag surrounding each input.
  232. public function display()
  233. {
  234. foreach ($this->widgets as $widget) {
  235. $star = $widget->isRequired() ? ' <span style="color:red">*</span>' : '';
  236. echo '<label>' . htmlspecialchars($widget->getLabel()) . $star . '<br/>';
  237. $widget->display();
  238. echo '</label><br/>';
  239. if (count($widget->errors)) {
  240. echo '<ul>';
  241. foreach ($widget->errors as $error) {
  242. echo '<li>' . htmlspecialchars($error) . '</li>';
  243. }
  244. echo '</ul>';
  245. }
  246. }
  247. }
  248. }
  249. ## Now let's put the whole thing to work...
  250. ##
  251. ## We'll create a `Person` object, create a `Form` for the object, and render it!
  252. ##
  253. ## Try leaving the name field empty, or try to tell the form you're 120 years old -
  254. ## it won't pass validation.
  255. ##
  256. ## You can see the state of the object being displayed below the form - as you can
  257. ## see, unless all updates and validations succeed, the state of your object is
  258. ## left untouched.
  259. echo <<<HTML
  260. <html>
  261. <head>
  262. <title>Metaprogramming With Annotations!</title>
  263. </head>
  264. <body>
  265. <h1>Edit a Person!</h1>
  266. <h4>Declarative Metaprogramming in action!</h4>
  267. <form method="post">
  268. HTML;
  269. $person = new Person;
  270. $form = new Form($person);
  271. if ($_SERVER['REQUEST_METHOD'] === 'POST') {
  272. if ($form->update($_POST)) {
  273. echo '<h2 style="color:green">Person Accepted!</h2>';
  274. } else {
  275. echo '<h2 style="color:red">Oops! Try again.</h2>';
  276. }
  277. }
  278. $form->display();
  279. echo <<<HTML
  280. <br/>
  281. <input type="submit" value="Go!"/>
  282. </form>
  283. HTML;
  284. echo "<pre>\n\nHere's what your Person instance currently looks like:\n\n";
  285. var_dump($person);
  286. echo '</pre>';
  287. echo <<<HTML
  288. </body>
  289. </html>
  290. HTML;