DemoScript.rst 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. Fully documented, step-by-step example of declarative meta-programming
  2. ======================================================================
  3. File: **demo/index.php**
  4. .. literalinclude:: ../demo/index.php
  5. :lines: 2-8
  6. Configure a simple auto-loader
  7. .. literalinclude:: ../demo/index.php
  8. :lines: 10-22
  9. Configure the cache-path. The static ``Annotations`` class will configure any public
  10. properties of ``AnnotationManager`` when it creates it. The ``AnnotationManager::$cachePath``
  11. property is a path to a writable folder, where the ``AnnotationManager`` caches parsed
  12. annotations from individual source code files.
  13. .. literalinclude:: ../demo/index.php
  14. :lines: 27-29
  15. Register demo annotations.
  16. .. literalinclude:: ../demo/index.php
  17. :lines: 31-32
  18. For this example, we're going to generate a simple form that allows us to edit a ``Person``
  19. object. We'll define a few public properties and annotate them with some useful metadata,
  20. which will enable us to make decisions (at run-time) about how to display each field,
  21. how to parse the values posted back from the form, and how to validate the input.
  22. Note the use of standard PHP-DOC annotations, such as ``@var string`` - this metadata is
  23. traditionally useful both as documentation to developers, and as hints for an IDE. In
  24. this example, we're going to use that same information as advice to our components, at
  25. run-time, to help them establish defaults and make sensible decisions about how to
  26. handle the value of each property.
  27. .. literalinclude:: ../demo/index.php
  28. :lines: 43-67
  29. To build a simple form abstraction that can manage the state of an object being edited,
  30. we start with a simple, abstract base class for input widgets.
  31. .. literalinclude:: ../demo/index.php
  32. :lines: 70-77
  33. Each widget will maintain a list of error messages.
  34. .. literalinclude:: ../demo/index.php
  35. :lines: 79-81
  36. A widget needs to know which property of what object is being edited.
  37. .. literalinclude:: ../demo/index.php
  38. :lines: 83-90
  39. Widget classes will use this method to add an error-message.
  40. .. literalinclude:: ../demo/index.php
  41. :lines: 92-97
  42. This helper function provides a shortcut to get a named property from a
  43. particular type of annotation - if no annotation is found, the ``$default``
  44. value is returned instead.
  45. .. literalinclude:: ../demo/index.php
  46. :lines: 101-112
  47. Each type of widget will need to implement this interface, which takes a raw
  48. POST value from the form, and attempts to bind it to the object's property.
  49. .. literalinclude:: ../demo/index.php
  50. :lines: 115-117
  51. After a widget successfully updates a property, we may need to perform additional
  52. validation - this method will perform some basic validations, and if errors are
  53. found, it will add them to the ``$errors`` collection.
  54. .. literalinclude:: ../demo/index.php
  55. :lines: 121-154
  56. Each type of widget will need to implement this interface, which renders an
  57. HTML input representing the widget's current value.
  58. .. literalinclude:: ../demo/index.php
  59. :lines: 157-159
  60. This helper function returns a descriptive label for the input.
  61. .. literalinclude:: ../demo/index.php
  62. :lines: 161-166
  63. Finally, this little helper function will tell us if the field is required -
  64. if a property is annotated with ``@required``, the field must be filled in.
  65. .. literalinclude:: ../demo/index.php
  66. :lines: 169-175
  67. The first and most basic kind of widget, is this simple string widget.
  68. .. literalinclude:: ../demo/index.php
  69. :lines: 177-179
  70. On update, take into account the min/max string length, and provide error
  71. messages if the constraints are violated.
  72. .. literalinclude:: ../demo/index.php
  73. :lines: 182-189
  74. On display, render out a simple ``<input type="text"/>`` field, taking into account
  75. the maximum string-length.
  76. .. literalinclude:: ../demo/index.php
  77. :lines: 192-201
  78. For the age input, we'll need a specialized ``StringWidget`` that also checks the input type.
  79. .. literalinclude:: ../demo/index.php
  80. :lines: 203-205
  81. On update, take into account the min/max numerical range, and provide error
  82. messages if the constraints are violated.
  83. .. literalinclude:: ../demo/index.php
  84. :lines: 208-223
  85. Next, we can build a simple form abstraction - this will hold and object and manage
  86. the widgets required to edit the object.
  87. .. literalinclude:: ../demo/index.php
  88. :lines: 226-237
  89. The constructor just needs to know which object we're editing.
  90. Using reflection, we enumerate the properties of the object's type, and using the
  91. ``@var`` annotation, we decide which type of widget we're going to use.
  92. .. literalinclude:: ../demo/index.php
  93. :lines: 242-257
  94. This helper-method is similar to the one we defined for the widget base
  95. class, but fetches annotations for the specified property.
  96. .. literalinclude:: ../demo/index.php
  97. :lines: 260-271
  98. When you post information back to the form, we'll need to update it's state,
  99. validate each of the fields, and return a value indicating whether the form
  100. update was successful.
  101. .. literalinclude:: ../demo/index.php
  102. :lines: 275-300
  103. Finally, this method renders out the form, and each of the widgets inside, with
  104. a ``<label>`` tag surrounding each input.
  105. .. literalinclude:: ../demo/index.php
  106. :lines: 303-322
  107. Now let's put the whole thing to work...
  108. We'll create a ``Person`` object, create a ``Form`` for the object, and render it!
  109. Try leaving the name field empty, or try to tell the form you're 120 years old -
  110. it won't pass validation.
  111. You can see the state of the object being displayed below the form - as you can
  112. see, unless all updates and validations succeed, the state of your object is
  113. left untouched.
  114. .. literalinclude:: ../demo/index.php
  115. :lines: 333-372